blob: 4f9a69688fec44f7d909761ced293f61a673d169 [file] [log] [blame]
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +00001//===-- NativeProcessNetBSD.cpp ------------------------------- -*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "NativeProcessNetBSD.h"
11
12// C Includes
13
14// C++ Includes
15
16// Other libraries and framework includes
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +000017#include "Plugins/Process/POSIX/ProcessPOSIXLog.h"
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000018#include "lldb/Core/State.h"
19#include "lldb/Host/HostProcess.h"
20#include "lldb/Host/common/NativeBreakpoint.h"
21#include "lldb/Host/common/NativeRegisterContext.h"
22#include "lldb/Host/posix/ProcessLauncherPosixFork.h"
23#include "lldb/Target/Process.h"
Pavel Labathc1a6b122017-07-03 09:25:55 +000024#include "llvm/Support/Errno.h"
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +000025
26// System includes - They have to be included after framework includes because
27// they define some
28// macros which collide with variable names in other modules
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000029// clang-format off
30#include <sys/types.h>
31#include <sys/ptrace.h>
32#include <sys/sysctl.h>
33#include <sys/wait.h>
34#include <uvm/uvm_prot.h>
35#include <elf.h>
36#include <util.h>
37// clang-format on
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +000038
39using namespace lldb;
40using namespace lldb_private;
41using namespace lldb_private::process_netbsd;
42using namespace llvm;
43
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000044// Simple helper function to ensure flags are enabled on the given file
45// descriptor.
Zachary Turner97206d52017-05-12 04:51:55 +000046static Status EnsureFDFlags(int fd, int flags) {
47 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000048
49 int status = fcntl(fd, F_GETFL);
50 if (status == -1) {
51 error.SetErrorToErrno();
52 return error;
53 }
54
55 if (fcntl(fd, F_SETFL, status | flags) == -1) {
56 error.SetErrorToErrno();
57 return error;
58 }
59
60 return error;
61}
62
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +000063// -----------------------------------------------------------------------------
64// Public Static Methods
65// -----------------------------------------------------------------------------
66
Pavel Labath82abefa2017-07-18 09:24:48 +000067llvm::Expected<std::unique_ptr<NativeProcessProtocol>>
Pavel Labath96e600f2017-07-07 11:02:19 +000068NativeProcessNetBSD::Factory::Launch(ProcessLaunchInfo &launch_info,
69 NativeDelegate &native_delegate,
70 MainLoop &mainloop) const {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000071 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
72
Pavel Labath96e600f2017-07-07 11:02:19 +000073 Status status;
74 ::pid_t pid = ProcessLauncherPosixFork()
75 .LaunchProcess(launch_info, status)
76 .GetProcessId();
77 LLDB_LOG(log, "pid = {0:x}", pid);
78 if (status.Fail()) {
79 LLDB_LOG(log, "failed to launch process: {0}", status);
80 return status.ToError();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000081 }
82
Pavel Labath96e600f2017-07-07 11:02:19 +000083 // Wait for the child process to trap on its call to execve.
84 int wstatus;
85 ::pid_t wpid = llvm::sys::RetryAfterSignal(-1, ::waitpid, pid, &wstatus, 0);
86 assert(wpid == pid);
87 (void)wpid;
88 if (!WIFSTOPPED(wstatus)) {
89 LLDB_LOG(log, "Could not sync with inferior process: wstatus={1}",
90 WaitStatus::Decode(wstatus));
91 return llvm::make_error<StringError>("Could not sync with inferior process",
92 llvm::inconvertibleErrorCode());
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000093 }
Pavel Labath96e600f2017-07-07 11:02:19 +000094 LLDB_LOG(log, "inferior started, now in stopped state");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000095
Pavel Labath96e600f2017-07-07 11:02:19 +000096 ArchSpec arch;
97 if ((status = ResolveProcessArchitecture(pid, arch)).Fail())
98 return status.ToError();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +000099
Pavel Labath96e600f2017-07-07 11:02:19 +0000100 // Set the architecture to the exe architecture.
101 LLDB_LOG(log, "pid = {0:x}, detected architecture {1}", pid,
102 arch.GetArchitectureName());
103
Pavel Labath82abefa2017-07-18 09:24:48 +0000104 std::unique_ptr<NativeProcessNetBSD> process_up(new NativeProcessNetBSD(
Pavel Labath96e600f2017-07-07 11:02:19 +0000105 pid, launch_info.GetPTY().ReleaseMasterFileDescriptor(), native_delegate,
106 arch, mainloop));
107
Pavel Labath82abefa2017-07-18 09:24:48 +0000108 status = process_up->ReinitializeThreads();
Pavel Labath96e600f2017-07-07 11:02:19 +0000109 if (status.Fail())
110 return status.ToError();
111
Pavel Labatha5be48b2017-10-17 15:52:16 +0000112 for (const auto &thread : process_up->m_threads)
113 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedBySignal(SIGSTOP);
Pavel Labath82abefa2017-07-18 09:24:48 +0000114 process_up->SetState(StateType::eStateStopped);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000115
Pavel Labath82abefa2017-07-18 09:24:48 +0000116 return std::move(process_up);
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +0000117}
118
Pavel Labath82abefa2017-07-18 09:24:48 +0000119llvm::Expected<std::unique_ptr<NativeProcessProtocol>>
120NativeProcessNetBSD::Factory::Attach(
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +0000121 lldb::pid_t pid, NativeProcessProtocol::NativeDelegate &native_delegate,
Pavel Labath96e600f2017-07-07 11:02:19 +0000122 MainLoop &mainloop) const {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000123 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
124 LLDB_LOG(log, "pid = {0:x}", pid);
125
126 // Retrieve the architecture for the running process.
Pavel Labath96e600f2017-07-07 11:02:19 +0000127 ArchSpec arch;
128 Status status = ResolveProcessArchitecture(pid, arch);
129 if (!status.Success())
130 return status.ToError();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000131
Pavel Labath82abefa2017-07-18 09:24:48 +0000132 std::unique_ptr<NativeProcessNetBSD> process_up(
Pavel Labath96e600f2017-07-07 11:02:19 +0000133 new NativeProcessNetBSD(pid, -1, native_delegate, arch, mainloop));
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000134
Pavel Labath82abefa2017-07-18 09:24:48 +0000135 status = process_up->Attach();
Pavel Labath96e600f2017-07-07 11:02:19 +0000136 if (!status.Success())
137 return status.ToError();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000138
Pavel Labath82abefa2017-07-18 09:24:48 +0000139 return std::move(process_up);
Kamil Rytarowski1a3d19d2017-03-21 17:30:47 +0000140}
141
142// -----------------------------------------------------------------------------
143// Public Instance Methods
144// -----------------------------------------------------------------------------
145
Pavel Labath96e600f2017-07-07 11:02:19 +0000146NativeProcessNetBSD::NativeProcessNetBSD(::pid_t pid, int terminal_fd,
147 NativeDelegate &delegate,
148 const ArchSpec &arch,
149 MainLoop &mainloop)
150 : NativeProcessProtocol(pid, terminal_fd, delegate), m_arch(arch) {
151 if (m_terminal_fd != -1) {
152 Status status = EnsureFDFlags(m_terminal_fd, O_NONBLOCK);
153 assert(status.Success());
154 }
155
156 Status status;
157 m_sigchld_handle = mainloop.RegisterSignal(
158 SIGCHLD, [this](MainLoopBase &) { SigchldHandler(); }, status);
159 assert(m_sigchld_handle && status.Success());
160}
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000161
162// Handles all waitpid events from the inferior process.
163void NativeProcessNetBSD::MonitorCallback(lldb::pid_t pid, int signal) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000164 switch (signal) {
165 case SIGTRAP:
166 return MonitorSIGTRAP(pid);
167 case SIGSTOP:
168 return MonitorSIGSTOP(pid);
169 default:
170 return MonitorSignal(pid, signal);
171 }
172}
173
Pavel Labath3508fc82017-06-19 12:47:50 +0000174void NativeProcessNetBSD::MonitorExited(lldb::pid_t pid, WaitStatus status) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000175 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
176
Pavel Labath3508fc82017-06-19 12:47:50 +0000177 LLDB_LOG(log, "got exit signal({0}) , pid = {1}", status, pid);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000178
179 /* Stop Tracking All Threads attached to Process */
180 m_threads.clear();
181
Pavel Labath3508fc82017-06-19 12:47:50 +0000182 SetExitStatus(status, true);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000183
184 // Notify delegate that our process has exited.
185 SetState(StateType::eStateExited, true);
186}
187
188void NativeProcessNetBSD::MonitorSIGSTOP(lldb::pid_t pid) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000189 ptrace_siginfo_t info;
190
191 const auto siginfo_err =
192 PtraceWrapper(PT_GET_SIGINFO, pid, &info, sizeof(info));
193
194 // Get details on the signal raised.
195 if (siginfo_err.Success()) {
196 // Handle SIGSTOP from LLGS (LLDB GDB Server)
197 if (info.psi_siginfo.si_code == SI_USER &&
198 info.psi_siginfo.si_pid == ::getpid()) {
Pavel Labatha5be48b2017-10-17 15:52:16 +0000199 /* Stop Tracking all Threads attached to Process */
200 for (const auto &thread : m_threads) {
201 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedBySignal(
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000202 SIGSTOP, &info.psi_siginfo);
203 }
204 }
205 }
206}
207
208void NativeProcessNetBSD::MonitorSIGTRAP(lldb::pid_t pid) {
209 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
210 ptrace_siginfo_t info;
211
212 const auto siginfo_err =
213 PtraceWrapper(PT_GET_SIGINFO, pid, &info, sizeof(info));
214
215 // Get details on the signal raised.
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000216 if (siginfo_err.Fail()) {
217 return;
218 }
219
220 switch (info.psi_siginfo.si_code) {
221 case TRAP_BRKPT:
Pavel Labatha5be48b2017-10-17 15:52:16 +0000222 for (const auto &thread : m_threads) {
223 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedByBreakpoint();
224 FixupBreakpointPCAsNeeded(static_cast<NativeThreadNetBSD &>(*thread));
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000225 }
226 SetState(StateType::eStateStopped, true);
227 break;
228 case TRAP_TRACE:
Pavel Labatha5be48b2017-10-17 15:52:16 +0000229 for (const auto &thread : m_threads)
230 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedByTrace();
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000231 SetState(StateType::eStateStopped, true);
232 break;
233 case TRAP_EXEC: {
Zachary Turner97206d52017-05-12 04:51:55 +0000234 Status error = ReinitializeThreads();
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000235 if (error.Fail()) {
236 SetState(StateType::eStateInvalid);
237 return;
238 }
239
240 // Let our delegate know we have just exec'd.
241 NotifyDidExec();
242
Pavel Labatha5be48b2017-10-17 15:52:16 +0000243 for (const auto &thread : m_threads)
244 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedByExec();
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000245 SetState(StateType::eStateStopped, true);
246 } break;
247 case TRAP_DBREG: {
248 // If a watchpoint was hit, report it
249 uint32_t wp_index;
Pavel Labatha5be48b2017-10-17 15:52:16 +0000250 Status error = static_cast<NativeThreadNetBSD &>(*m_threads[info.psi_lwpid])
251 .GetRegisterContext()
252 ->GetWatchpointHitIndex(
253 wp_index, (uintptr_t)info.psi_siginfo.si_addr);
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000254 if (error.Fail())
255 LLDB_LOG(log,
256 "received error while checking for watchpoint hits, pid = "
257 "{0}, LWP = {1}, error = {2}",
258 GetID(), info.psi_lwpid, error);
259 if (wp_index != LLDB_INVALID_INDEX32) {
Pavel Labatha5be48b2017-10-17 15:52:16 +0000260 for (const auto &thread : m_threads)
261 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedByWatchpoint(
262 wp_index);
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000263 SetState(StateType::eStateStopped, true);
264 break;
265 }
266
267 // If a breakpoint was hit, report it
268 uint32_t bp_index;
Pavel Labatha5be48b2017-10-17 15:52:16 +0000269 error = static_cast<NativeThreadNetBSD &>(*m_threads[info.psi_lwpid])
270 .GetRegisterContext()
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000271 ->GetHardwareBreakHitIndex(bp_index,
272 (uintptr_t)info.psi_siginfo.si_addr);
273 if (error.Fail())
274 LLDB_LOG(log,
275 "received error while checking for hardware "
276 "breakpoint hits, pid = {0}, LWP = {1}, error = {2}",
277 GetID(), info.psi_lwpid, error);
278 if (bp_index != LLDB_INVALID_INDEX32) {
Pavel Labatha5be48b2017-10-17 15:52:16 +0000279 for (const auto &thread : m_threads)
280 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedByBreakpoint();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000281 SetState(StateType::eStateStopped, true);
282 break;
283 }
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000284 } break;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000285 }
286}
287
288void NativeProcessNetBSD::MonitorSignal(lldb::pid_t pid, int signal) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000289 ptrace_siginfo_t info;
290 const auto siginfo_err =
291 PtraceWrapper(PT_GET_SIGINFO, pid, &info, sizeof(info));
292
Pavel Labatha5be48b2017-10-17 15:52:16 +0000293 for (const auto &thread : m_threads) {
294 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedBySignal(
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000295 info.psi_siginfo.si_signo, &info.psi_siginfo);
296 }
297 SetState(StateType::eStateStopped, true);
298}
299
Zachary Turner97206d52017-05-12 04:51:55 +0000300Status NativeProcessNetBSD::PtraceWrapper(int req, lldb::pid_t pid, void *addr,
301 int data, int *result) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000302 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PTRACE));
Zachary Turner97206d52017-05-12 04:51:55 +0000303 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000304 int ret;
305
306 errno = 0;
307 ret = ptrace(req, static_cast<::pid_t>(pid), addr, data);
308
309 if (ret == -1)
310 error.SetErrorToErrno();
311
312 if (result)
313 *result = ret;
314
315 LLDB_LOG(log, "ptrace({0}, {1}, {2}, {3})={4:x}", req, pid, addr, data, ret);
316
317 if (error.Fail())
318 LLDB_LOG(log, "ptrace() failed: {0}", error);
319
320 return error;
321}
322
Zachary Turner97206d52017-05-12 04:51:55 +0000323Status NativeProcessNetBSD::GetSoftwareBreakpointPCOffset(
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000324 uint32_t &actual_opcode_size) {
325 // FIXME put this behind a breakpoint protocol class that can be
326 // set per architecture. Need ARM, MIPS support here.
327 static const uint8_t g_i386_opcode[] = {0xCC};
328 switch (m_arch.GetMachine()) {
329 case llvm::Triple::x86_64:
330 actual_opcode_size = static_cast<uint32_t>(sizeof(g_i386_opcode));
Zachary Turner97206d52017-05-12 04:51:55 +0000331 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000332 default:
333 assert(false && "CPU type not supported!");
Zachary Turner97206d52017-05-12 04:51:55 +0000334 return Status("CPU type not supported");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000335 }
336}
337
Zachary Turner97206d52017-05-12 04:51:55 +0000338Status
339NativeProcessNetBSD::FixupBreakpointPCAsNeeded(NativeThreadNetBSD &thread) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000340 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_BREAKPOINTS));
Zachary Turner97206d52017-05-12 04:51:55 +0000341 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000342 // Find out the size of a breakpoint (might depend on where we are in the
343 // code).
344 NativeRegisterContextSP context_sp = thread.GetRegisterContext();
345 if (!context_sp) {
346 error.SetErrorString("cannot get a NativeRegisterContext for the thread");
347 LLDB_LOG(log, "failed: {0}", error);
348 return error;
349 }
350 uint32_t breakpoint_size = 0;
351 error = GetSoftwareBreakpointPCOffset(breakpoint_size);
352 if (error.Fail()) {
353 LLDB_LOG(log, "GetBreakpointSize() failed: {0}", error);
354 return error;
355 } else
356 LLDB_LOG(log, "breakpoint size: {0}", breakpoint_size);
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000357 // First try probing for a breakpoint at a software breakpoint location: PC
358 // - breakpoint size.
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000359 const lldb::addr_t initial_pc_addr =
360 context_sp->GetPCfromBreakpointLocation();
361 lldb::addr_t breakpoint_addr = initial_pc_addr;
362 if (breakpoint_size > 0) {
363 // Do not allow breakpoint probe to wrap around.
364 if (breakpoint_addr >= breakpoint_size)
365 breakpoint_addr -= breakpoint_size;
366 }
367 // Check if we stopped because of a breakpoint.
368 NativeBreakpointSP breakpoint_sp;
369 error = m_breakpoint_list.GetBreakpoint(breakpoint_addr, breakpoint_sp);
370 if (!error.Success() || !breakpoint_sp) {
371 // We didn't find one at a software probe location. Nothing to do.
372 LLDB_LOG(log,
373 "pid {0} no lldb breakpoint found at current pc with "
374 "adjustment: {1}",
375 GetID(), breakpoint_addr);
Zachary Turner97206d52017-05-12 04:51:55 +0000376 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000377 }
378 // If the breakpoint is not a software breakpoint, nothing to do.
379 if (!breakpoint_sp->IsSoftwareBreakpoint()) {
380 LLDB_LOG(
381 log,
382 "pid {0} breakpoint found at {1:x}, not software, nothing to adjust",
383 GetID(), breakpoint_addr);
Zachary Turner97206d52017-05-12 04:51:55 +0000384 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000385 }
386 //
387 // We have a software breakpoint and need to adjust the PC.
388 //
389 // Sanity check.
390 if (breakpoint_size == 0) {
391 // Nothing to do! How did we get here?
392 LLDB_LOG(log,
393 "pid {0} breakpoint found at {1:x}, it is software, but the "
394 "size is zero, nothing to do (unexpected)",
395 GetID(), breakpoint_addr);
Zachary Turner97206d52017-05-12 04:51:55 +0000396 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000397 }
398 //
399 // We have a software breakpoint and need to adjust the PC.
400 //
401 // Sanity check.
402 if (breakpoint_size == 0) {
403 // Nothing to do! How did we get here?
404 LLDB_LOG(log,
405 "pid {0} breakpoint found at {1:x}, it is software, but the "
406 "size is zero, nothing to do (unexpected)",
407 GetID(), breakpoint_addr);
Zachary Turner97206d52017-05-12 04:51:55 +0000408 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000409 }
410 // Change the program counter.
411 LLDB_LOG(log, "pid {0} tid {1}: changing PC from {2:x} to {3:x}", GetID(),
412 thread.GetID(), initial_pc_addr, breakpoint_addr);
413 error = context_sp->SetPC(breakpoint_addr);
414 if (error.Fail()) {
415 LLDB_LOG(log, "pid {0} tid {1}: failed to set PC: {2}", GetID(),
416 thread.GetID(), error);
417 return error;
418 }
419 return error;
420}
421
Zachary Turner97206d52017-05-12 04:51:55 +0000422Status NativeProcessNetBSD::Resume(const ResumeActionList &resume_actions) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000423 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
424 LLDB_LOG(log, "pid {0}", GetID());
425
Pavel Labatha5be48b2017-10-17 15:52:16 +0000426 const auto &thread = m_threads[0];
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000427 const ResumeAction *const action =
Pavel Labatha5be48b2017-10-17 15:52:16 +0000428 resume_actions.GetActionForThread(thread->GetID(), true);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000429
430 if (action == nullptr) {
431 LLDB_LOG(log, "no action specified for pid {0} tid {1}", GetID(),
Pavel Labatha5be48b2017-10-17 15:52:16 +0000432 thread->GetID());
Zachary Turner97206d52017-05-12 04:51:55 +0000433 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000434 }
435
Zachary Turner97206d52017-05-12 04:51:55 +0000436 Status error;
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000437
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000438 switch (action->state) {
439 case eStateRunning: {
440 // Run the thread, possibly feeding it the signal.
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000441 error = NativeProcessNetBSD::PtraceWrapper(PT_CONTINUE, GetID(), (void *)1,
442 action->signal);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000443 if (!error.Success())
444 return error;
Pavel Labatha5be48b2017-10-17 15:52:16 +0000445 for (const auto &thread : m_threads)
446 static_cast<NativeThreadNetBSD &>(*thread).SetRunning();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000447 SetState(eStateRunning, true);
448 break;
449 }
450 case eStateStepping:
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000451 // Run the thread, possibly feeding it the signal.
452 error = NativeProcessNetBSD::PtraceWrapper(PT_STEP, GetID(), (void *)1,
453 action->signal);
454 if (!error.Success())
455 return error;
Pavel Labatha5be48b2017-10-17 15:52:16 +0000456 for (const auto &thread : m_threads)
457 static_cast<NativeThreadNetBSD &>(*thread).SetStepping();
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000458 SetState(eStateStepping, true);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000459 break;
460
461 case eStateSuspended:
462 case eStateStopped:
463 llvm_unreachable("Unexpected state");
464
465 default:
Zachary Turner97206d52017-05-12 04:51:55 +0000466 return Status("NativeProcessNetBSD::%s (): unexpected state %s specified "
467 "for pid %" PRIu64 ", tid %" PRIu64,
468 __FUNCTION__, StateAsCString(action->state), GetID(),
Pavel Labatha5be48b2017-10-17 15:52:16 +0000469 thread->GetID());
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000470 }
471
Zachary Turner97206d52017-05-12 04:51:55 +0000472 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000473}
474
Zachary Turner97206d52017-05-12 04:51:55 +0000475Status NativeProcessNetBSD::Halt() {
476 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000477
478 if (kill(GetID(), SIGSTOP) != 0)
479 error.SetErrorToErrno();
480
481 return error;
482}
483
Zachary Turner97206d52017-05-12 04:51:55 +0000484Status NativeProcessNetBSD::Detach() {
485 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000486
487 // Stop monitoring the inferior.
488 m_sigchld_handle.reset();
489
490 // Tell ptrace to detach from the process.
491 if (GetID() == LLDB_INVALID_PROCESS_ID)
492 return error;
493
494 return PtraceWrapper(PT_DETACH, GetID());
495}
496
Zachary Turner97206d52017-05-12 04:51:55 +0000497Status NativeProcessNetBSD::Signal(int signo) {
498 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000499
500 if (kill(GetID(), signo))
501 error.SetErrorToErrno();
502
503 return error;
504}
505
Zachary Turner97206d52017-05-12 04:51:55 +0000506Status NativeProcessNetBSD::Kill() {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000507 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
508 LLDB_LOG(log, "pid {0}", GetID());
509
Zachary Turner97206d52017-05-12 04:51:55 +0000510 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000511
512 switch (m_state) {
513 case StateType::eStateInvalid:
514 case StateType::eStateExited:
515 case StateType::eStateCrashed:
516 case StateType::eStateDetached:
517 case StateType::eStateUnloaded:
518 // Nothing to do - the process is already dead.
519 LLDB_LOG(log, "ignored for PID {0} due to current state: {1}", GetID(),
520 StateAsCString(m_state));
521 return error;
522
523 case StateType::eStateConnected:
524 case StateType::eStateAttaching:
525 case StateType::eStateLaunching:
526 case StateType::eStateStopped:
527 case StateType::eStateRunning:
528 case StateType::eStateStepping:
529 case StateType::eStateSuspended:
530 // We can try to kill a process in these states.
531 break;
532 }
533
534 if (kill(GetID(), SIGKILL) != 0) {
535 error.SetErrorToErrno();
536 return error;
537 }
538
539 return error;
540}
541
Zachary Turner97206d52017-05-12 04:51:55 +0000542Status NativeProcessNetBSD::GetMemoryRegionInfo(lldb::addr_t load_addr,
543 MemoryRegionInfo &range_info) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000544
545 if (m_supports_mem_region == LazyBool::eLazyBoolNo) {
546 // We're done.
Zachary Turner97206d52017-05-12 04:51:55 +0000547 return Status("unsupported");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000548 }
549
Zachary Turner97206d52017-05-12 04:51:55 +0000550 Status error = PopulateMemoryRegionCache();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000551 if (error.Fail()) {
552 return error;
553 }
554
555 lldb::addr_t prev_base_address = 0;
556 // FIXME start by finding the last region that is <= target address using
557 // binary search. Data is sorted.
558 // There can be a ton of regions on pthreads apps with lots of threads.
559 for (auto it = m_mem_region_cache.begin(); it != m_mem_region_cache.end();
560 ++it) {
561 MemoryRegionInfo &proc_entry_info = it->first;
562 // Sanity check assumption that memory map entries are ascending.
563 assert((proc_entry_info.GetRange().GetRangeBase() >= prev_base_address) &&
564 "descending memory map entries detected, unexpected");
565 prev_base_address = proc_entry_info.GetRange().GetRangeBase();
566 UNUSED_IF_ASSERT_DISABLED(prev_base_address);
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000567 // If the target address comes before this entry, indicate distance to
568 // next region.
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000569 if (load_addr < proc_entry_info.GetRange().GetRangeBase()) {
570 range_info.GetRange().SetRangeBase(load_addr);
571 range_info.GetRange().SetByteSize(
572 proc_entry_info.GetRange().GetRangeBase() - load_addr);
573 range_info.SetReadable(MemoryRegionInfo::OptionalBool::eNo);
574 range_info.SetWritable(MemoryRegionInfo::OptionalBool::eNo);
575 range_info.SetExecutable(MemoryRegionInfo::OptionalBool::eNo);
576 range_info.SetMapped(MemoryRegionInfo::OptionalBool::eNo);
577 return error;
578 } else if (proc_entry_info.GetRange().Contains(load_addr)) {
579 // The target address is within the memory region we're processing here.
580 range_info = proc_entry_info;
581 return error;
582 }
583 // The target memory address comes somewhere after the region we just
584 // parsed.
585 }
586 // If we made it here, we didn't find an entry that contained the given
587 // address. Return the
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000588 // load_addr as start and the amount of bytes betwwen load address and the
589 // end of the memory as size.
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000590 range_info.GetRange().SetRangeBase(load_addr);
591 range_info.GetRange().SetRangeEnd(LLDB_INVALID_ADDRESS);
592 range_info.SetReadable(MemoryRegionInfo::OptionalBool::eNo);
593 range_info.SetWritable(MemoryRegionInfo::OptionalBool::eNo);
594 range_info.SetExecutable(MemoryRegionInfo::OptionalBool::eNo);
595 range_info.SetMapped(MemoryRegionInfo::OptionalBool::eNo);
596 return error;
597}
598
Zachary Turner97206d52017-05-12 04:51:55 +0000599Status NativeProcessNetBSD::PopulateMemoryRegionCache() {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000600 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
601 // If our cache is empty, pull the latest. There should always be at least
602 // one memory region if memory region handling is supported.
603 if (!m_mem_region_cache.empty()) {
604 LLDB_LOG(log, "reusing {0} cached memory region entries",
605 m_mem_region_cache.size());
Zachary Turner97206d52017-05-12 04:51:55 +0000606 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000607 }
608
609 struct kinfo_vmentry *vm;
610 size_t count, i;
611 vm = kinfo_getvmmap(GetID(), &count);
612 if (vm == NULL) {
613 m_supports_mem_region = LazyBool::eLazyBoolNo;
Zachary Turner97206d52017-05-12 04:51:55 +0000614 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000615 error.SetErrorString("not supported");
616 return error;
617 }
618 for (i = 0; i < count; i++) {
619 MemoryRegionInfo info;
620 info.Clear();
621 info.GetRange().SetRangeBase(vm[i].kve_start);
622 info.GetRange().SetRangeEnd(vm[i].kve_end);
623 info.SetMapped(MemoryRegionInfo::OptionalBool::eYes);
624
625 if (vm[i].kve_protection & VM_PROT_READ)
626 info.SetReadable(MemoryRegionInfo::OptionalBool::eYes);
627 else
628 info.SetReadable(MemoryRegionInfo::OptionalBool::eNo);
629
630 if (vm[i].kve_protection & VM_PROT_WRITE)
631 info.SetWritable(MemoryRegionInfo::OptionalBool::eYes);
632 else
633 info.SetWritable(MemoryRegionInfo::OptionalBool::eNo);
634
635 if (vm[i].kve_protection & VM_PROT_EXECUTE)
636 info.SetExecutable(MemoryRegionInfo::OptionalBool::eYes);
637 else
638 info.SetExecutable(MemoryRegionInfo::OptionalBool::eNo);
639
640 if (vm[i].kve_path[0])
641 info.SetName(vm[i].kve_path);
642
643 m_mem_region_cache.emplace_back(
644 info, FileSpec(info.GetName().GetCString(), true));
645 }
646 free(vm);
647
648 if (m_mem_region_cache.empty()) {
649 // No entries after attempting to read them. This shouldn't happen.
650 // Assume we don't support map entries.
651 LLDB_LOG(log, "failed to find any vmmap entries, assuming no support "
652 "for memory region metadata retrieval");
653 m_supports_mem_region = LazyBool::eLazyBoolNo;
Zachary Turner97206d52017-05-12 04:51:55 +0000654 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000655 error.SetErrorString("not supported");
656 return error;
657 }
658 LLDB_LOG(log, "read {0} memory region entries from process {1}",
659 m_mem_region_cache.size(), GetID());
660 // We support memory retrieval, remember that.
661 m_supports_mem_region = LazyBool::eLazyBoolYes;
Zachary Turner97206d52017-05-12 04:51:55 +0000662 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000663}
664
Zachary Turner97206d52017-05-12 04:51:55 +0000665Status NativeProcessNetBSD::AllocateMemory(size_t size, uint32_t permissions,
666 lldb::addr_t &addr) {
667 return Status("Unimplemented");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000668}
669
Zachary Turner97206d52017-05-12 04:51:55 +0000670Status NativeProcessNetBSD::DeallocateMemory(lldb::addr_t addr) {
671 return Status("Unimplemented");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000672}
673
674lldb::addr_t NativeProcessNetBSD::GetSharedLibraryInfoAddress() {
675 // punt on this for now
676 return LLDB_INVALID_ADDRESS;
677}
678
679size_t NativeProcessNetBSD::UpdateThreads() { return m_threads.size(); }
680
681bool NativeProcessNetBSD::GetArchitecture(ArchSpec &arch) const {
682 arch = m_arch;
683 return true;
684}
685
Zachary Turner97206d52017-05-12 04:51:55 +0000686Status NativeProcessNetBSD::SetBreakpoint(lldb::addr_t addr, uint32_t size,
687 bool hardware) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000688 if (hardware)
Zachary Turner97206d52017-05-12 04:51:55 +0000689 return Status("NativeProcessNetBSD does not support hardware breakpoints");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000690 else
691 return SetSoftwareBreakpoint(addr, size);
692}
693
Zachary Turner97206d52017-05-12 04:51:55 +0000694Status NativeProcessNetBSD::GetSoftwareBreakpointTrapOpcode(
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000695 size_t trap_opcode_size_hint, size_t &actual_opcode_size,
696 const uint8_t *&trap_opcode_bytes) {
697 static const uint8_t g_i386_opcode[] = {0xCC};
698
699 switch (m_arch.GetMachine()) {
700 case llvm::Triple::x86:
701 case llvm::Triple::x86_64:
702 trap_opcode_bytes = g_i386_opcode;
703 actual_opcode_size = sizeof(g_i386_opcode);
Zachary Turner97206d52017-05-12 04:51:55 +0000704 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000705 default:
706 assert(false && "CPU type not supported!");
Zachary Turner97206d52017-05-12 04:51:55 +0000707 return Status("CPU type not supported");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000708 }
709}
710
Zachary Turner97206d52017-05-12 04:51:55 +0000711Status NativeProcessNetBSD::GetLoadedModuleFileSpec(const char *module_path,
712 FileSpec &file_spec) {
713 return Status("Unimplemented");
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000714}
715
Zachary Turner97206d52017-05-12 04:51:55 +0000716Status NativeProcessNetBSD::GetFileLoadAddress(const llvm::StringRef &file_name,
717 lldb::addr_t &load_addr) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000718 load_addr = LLDB_INVALID_ADDRESS;
Zachary Turner97206d52017-05-12 04:51:55 +0000719 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000720}
721
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000722void NativeProcessNetBSD::SigchldHandler() {
723 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_PROCESS));
724 // Process all pending waitpid notifications.
725 int status;
Pavel Labathc1a6b122017-07-03 09:25:55 +0000726 ::pid_t wait_pid =
727 llvm::sys::RetryAfterSignal(-1, waitpid, GetID(), &status, WALLSIG | WNOHANG);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000728
729 if (wait_pid == 0)
730 return; // We are done.
731
732 if (wait_pid == -1) {
Zachary Turner97206d52017-05-12 04:51:55 +0000733 Status error(errno, eErrorTypePOSIX);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000734 LLDB_LOG(log, "waitpid ({0}, &status, _) failed: {1}", GetID(), error);
735 }
736
Pavel Labath3508fc82017-06-19 12:47:50 +0000737 WaitStatus wait_status = WaitStatus::Decode(status);
738 bool exited = wait_status.type == WaitStatus::Exit ||
739 (wait_status.type == WaitStatus::Signal &&
740 wait_pid == static_cast<::pid_t>(GetID()));
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000741
742 LLDB_LOG(log,
Pavel Labath3508fc82017-06-19 12:47:50 +0000743 "waitpid ({0}, &status, _) => pid = {1}, status = {2}, exited = {3}",
744 GetID(), wait_pid, status, exited);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000745
746 if (exited)
Pavel Labath3508fc82017-06-19 12:47:50 +0000747 MonitorExited(wait_pid, wait_status);
748 else {
Kamil Rytarowski4bb74412017-06-20 13:51:06 +0000749 assert(wait_status.type == WaitStatus::Stop);
Pavel Labath3508fc82017-06-19 12:47:50 +0000750 MonitorCallback(wait_pid, wait_status.status);
751 }
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000752}
753
Kamil Rytarowski269eec02017-05-24 23:59:50 +0000754bool NativeProcessNetBSD::HasThreadNoLock(lldb::tid_t thread_id) {
Pavel Labatha5be48b2017-10-17 15:52:16 +0000755 for (const auto &thread : m_threads) {
756 assert(thread && "thread list should not contain NULL threads");
757 if (thread->GetID() == thread_id) {
Kamil Rytarowski269eec02017-05-24 23:59:50 +0000758 // We have this thread.
759 return true;
760 }
761 }
762
763 // We don't have this thread.
764 return false;
765}
766
Pavel Labatha5be48b2017-10-17 15:52:16 +0000767NativeThreadNetBSD &NativeProcessNetBSD::AddThread(lldb::tid_t thread_id) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000768
769 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_THREAD));
770 LLDB_LOG(log, "pid {0} adding thread with tid {1}", GetID(), thread_id);
771
772 assert(!HasThreadNoLock(thread_id) &&
773 "attempted to add a thread by id that already exists");
774
775 // If this is the first thread, save it as the current thread
776 if (m_threads.empty())
777 SetCurrentThreadID(thread_id);
778
Pavel Labatha5be48b2017-10-17 15:52:16 +0000779 m_threads.push_back(llvm::make_unique<NativeThreadNetBSD>(*this, thread_id));
780 return static_cast<NativeThreadNetBSD &>(*m_threads.back());
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000781}
782
Pavel Labath96e600f2017-07-07 11:02:19 +0000783Status NativeProcessNetBSD::Attach() {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000784 // Attach to the requested process.
785 // An attach will cause the thread to stop with a SIGSTOP.
Pavel Labath96e600f2017-07-07 11:02:19 +0000786 Status status = PtraceWrapper(PT_ATTACH, m_pid);
787 if (status.Fail())
788 return status;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000789
Pavel Labath96e600f2017-07-07 11:02:19 +0000790 int wstatus;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000791 // Need to use WALLSIG otherwise we receive an error with errno=ECHLD
792 // At this point we should have a thread stopped if waitpid succeeds.
Pavel Labath96e600f2017-07-07 11:02:19 +0000793 if ((wstatus = waitpid(m_pid, NULL, WALLSIG)) < 0)
794 return Status(errno, eErrorTypePOSIX);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000795
796 /* Initialize threads */
Pavel Labath96e600f2017-07-07 11:02:19 +0000797 status = ReinitializeThreads();
798 if (status.Fail())
799 return status;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000800
Pavel Labatha5be48b2017-10-17 15:52:16 +0000801 for (const auto &thread : m_threads)
802 static_cast<NativeThreadNetBSD &>(*thread).SetStoppedBySignal(SIGSTOP);
Kamil Rytarowski36e23ec2017-04-18 12:53:35 +0000803
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000804 // Let our process instance know the thread has stopped.
805 SetState(StateType::eStateStopped);
Pavel Labath96e600f2017-07-07 11:02:19 +0000806 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000807}
808
Zachary Turner97206d52017-05-12 04:51:55 +0000809Status NativeProcessNetBSD::ReadMemory(lldb::addr_t addr, void *buf,
810 size_t size, size_t &bytes_read) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000811 unsigned char *dst = static_cast<unsigned char *>(buf);
812 struct ptrace_io_desc io;
813
814 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_MEMORY));
815 LLDB_LOG(log, "addr = {0}, buf = {1}, size = {2}", addr, buf, size);
816
817 bytes_read = 0;
818 io.piod_op = PIOD_READ_D;
819 io.piod_len = size;
820
821 do {
822 io.piod_offs = (void *)(addr + bytes_read);
823 io.piod_addr = dst + bytes_read;
824
Zachary Turner97206d52017-05-12 04:51:55 +0000825 Status error = NativeProcessNetBSD::PtraceWrapper(PT_IO, GetID(), &io);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000826 if (error.Fail())
827 return error;
828
829 bytes_read = io.piod_len;
830 io.piod_len = size - bytes_read;
831 } while (bytes_read < size);
832
Zachary Turner97206d52017-05-12 04:51:55 +0000833 return Status();
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000834}
835
Zachary Turner97206d52017-05-12 04:51:55 +0000836Status NativeProcessNetBSD::ReadMemoryWithoutTrap(lldb::addr_t addr, void *buf,
837 size_t size,
838 size_t &bytes_read) {
839 Status error = ReadMemory(addr, buf, size, bytes_read);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000840 if (error.Fail())
841 return error;
842 return m_breakpoint_list.RemoveTrapsFromBuffer(addr, buf, size);
843}
844
Zachary Turner97206d52017-05-12 04:51:55 +0000845Status NativeProcessNetBSD::WriteMemory(lldb::addr_t addr, const void *buf,
846 size_t size, size_t &bytes_written) {
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000847 const unsigned char *src = static_cast<const unsigned char *>(buf);
Zachary Turner97206d52017-05-12 04:51:55 +0000848 Status error;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000849 struct ptrace_io_desc io;
850
851 Log *log(ProcessPOSIXLog::GetLogIfAllCategoriesSet(POSIX_LOG_MEMORY));
852 LLDB_LOG(log, "addr = {0}, buf = {1}, size = {2}", addr, buf, size);
853
854 bytes_written = 0;
855 io.piod_op = PIOD_WRITE_D;
856 io.piod_len = size;
857
858 do {
Kamil Rytarowski269eec02017-05-24 23:59:50 +0000859 io.piod_addr = const_cast<void *>(static_cast<const void *>(src + bytes_written));
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000860 io.piod_offs = (void *)(addr + bytes_written);
861
Zachary Turner97206d52017-05-12 04:51:55 +0000862 Status error = NativeProcessNetBSD::PtraceWrapper(PT_IO, GetID(), &io);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000863 if (error.Fail())
864 return error;
865
866 bytes_written = io.piod_len;
867 io.piod_len = size - bytes_written;
868 } while (bytes_written < size);
869
870 return error;
871}
872
873llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>>
874NativeProcessNetBSD::GetAuxvData() const {
875 /*
876 * ELF_AUX_ENTRIES is currently restricted to kernel
877 * (<sys/exec_elf.h> r. 1.155 specifies 15)
878 *
879 * ptrace(2) returns the whole AUXV including extra fiels after AT_NULL this
880 * information isn't needed.
881 */
882 size_t auxv_size = 100 * sizeof(AuxInfo);
883
884 ErrorOr<std::unique_ptr<MemoryBuffer>> buf =
885 llvm::MemoryBuffer::getNewMemBuffer(auxv_size);
886
Kamil Rytarowski269eec02017-05-24 23:59:50 +0000887 struct ptrace_io_desc io;
888 io.piod_op = PIOD_READ_AUXV;
889 io.piod_offs = 0;
890 io.piod_addr = const_cast<void *>(static_cast<const void *>(buf.get()->getBufferStart()));
891 io.piod_len = auxv_size;
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000892
Zachary Turner97206d52017-05-12 04:51:55 +0000893 Status error = NativeProcessNetBSD::PtraceWrapper(PT_IO, GetID(), &io);
Kamil Rytarowskif07a9992017-03-28 22:43:17 +0000894
895 if (error.Fail())
896 return std::error_code(error.GetError(), std::generic_category());
897
898 if (io.piod_len < 1)
899 return std::error_code(ECANCELED, std::generic_category());
900
901 return buf;
902}
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000903
Zachary Turner97206d52017-05-12 04:51:55 +0000904Status NativeProcessNetBSD::ReinitializeThreads() {
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000905 // Clear old threads
906 m_threads.clear();
907
908 // Initialize new thread
909 struct ptrace_lwpinfo info = {};
Zachary Turner97206d52017-05-12 04:51:55 +0000910 Status error = PtraceWrapper(PT_LWPINFO, GetID(), &info, sizeof(info));
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000911 if (error.Fail()) {
912 return error;
913 }
914 // Reinitialize from scratch threads and register them in process
915 while (info.pl_lwpid != 0) {
Pavel Labatha5be48b2017-10-17 15:52:16 +0000916 AddThread(info.pl_lwpid);
Kamil Rytarowski3eef2b52017-03-30 20:25:29 +0000917 error = PtraceWrapper(PT_LWPINFO, GetID(), &info, sizeof(info));
918 if (error.Fail()) {
919 return error;
920 }
921 }
922
923 return error;
924}