forked from OSchip/llvm-project
1477 lines
50 KiB
C++
1477 lines
50 KiB
C++
//===-- SBProcess.cpp -----------------------------------------------------===//
|
|
//
|
|
// 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 "lldb/API/SBProcess.h"
|
|
#include "SBReproducerPrivate.h"
|
|
|
|
#include <cinttypes>
|
|
|
|
#include "lldb/lldb-defines.h"
|
|
#include "lldb/lldb-types.h"
|
|
|
|
#include "lldb/Core/Debugger.h"
|
|
#include "lldb/Core/Module.h"
|
|
#include "lldb/Core/PluginManager.h"
|
|
#include "lldb/Core/StreamFile.h"
|
|
#include "lldb/Core/StructuredDataImpl.h"
|
|
#include "lldb/Target/MemoryRegionInfo.h"
|
|
#include "lldb/Target/Process.h"
|
|
#include "lldb/Target/RegisterContext.h"
|
|
#include "lldb/Target/SystemRuntime.h"
|
|
#include "lldb/Target/Target.h"
|
|
#include "lldb/Target/Thread.h"
|
|
#include "lldb/Utility/Args.h"
|
|
#include "lldb/Utility/ProcessInfo.h"
|
|
#include "lldb/Utility/State.h"
|
|
#include "lldb/Utility/Stream.h"
|
|
|
|
#include "lldb/API/SBBroadcaster.h"
|
|
#include "lldb/API/SBCommandReturnObject.h"
|
|
#include "lldb/API/SBDebugger.h"
|
|
#include "lldb/API/SBEvent.h"
|
|
#include "lldb/API/SBFile.h"
|
|
#include "lldb/API/SBFileSpec.h"
|
|
#include "lldb/API/SBMemoryRegionInfo.h"
|
|
#include "lldb/API/SBMemoryRegionInfoList.h"
|
|
#include "lldb/API/SBStream.h"
|
|
#include "lldb/API/SBStringList.h"
|
|
#include "lldb/API/SBStructuredData.h"
|
|
#include "lldb/API/SBThread.h"
|
|
#include "lldb/API/SBThreadCollection.h"
|
|
#include "lldb/API/SBTrace.h"
|
|
#include "lldb/API/SBUnixSignals.h"
|
|
|
|
using namespace lldb;
|
|
using namespace lldb_private;
|
|
|
|
SBProcess::SBProcess() : m_opaque_wp() {
|
|
LLDB_RECORD_CONSTRUCTOR_NO_ARGS(SBProcess);
|
|
}
|
|
|
|
// SBProcess constructor
|
|
|
|
SBProcess::SBProcess(const SBProcess &rhs) : m_opaque_wp(rhs.m_opaque_wp) {
|
|
LLDB_RECORD_CONSTRUCTOR(SBProcess, (const lldb::SBProcess &), rhs);
|
|
}
|
|
|
|
SBProcess::SBProcess(const lldb::ProcessSP &process_sp)
|
|
: m_opaque_wp(process_sp) {
|
|
LLDB_RECORD_CONSTRUCTOR(SBProcess, (const lldb::ProcessSP &), process_sp);
|
|
}
|
|
|
|
const SBProcess &SBProcess::operator=(const SBProcess &rhs) {
|
|
LLDB_RECORD_METHOD(const lldb::SBProcess &,
|
|
SBProcess, operator=,(const lldb::SBProcess &), rhs);
|
|
|
|
if (this != &rhs)
|
|
m_opaque_wp = rhs.m_opaque_wp;
|
|
return LLDB_RECORD_RESULT(*this);
|
|
}
|
|
|
|
// Destructor
|
|
SBProcess::~SBProcess() = default;
|
|
|
|
const char *SBProcess::GetBroadcasterClassName() {
|
|
LLDB_RECORD_STATIC_METHOD_NO_ARGS(const char *, SBProcess,
|
|
GetBroadcasterClassName);
|
|
|
|
return Process::GetStaticBroadcasterClass().AsCString();
|
|
}
|
|
|
|
const char *SBProcess::GetPluginName() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(const char *, SBProcess, GetPluginName);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
return process_sp->GetPluginName().GetCString();
|
|
}
|
|
return "<Unknown>";
|
|
}
|
|
|
|
const char *SBProcess::GetShortPluginName() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(const char *, SBProcess, GetShortPluginName);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
return process_sp->GetPluginName().GetCString();
|
|
}
|
|
return "<Unknown>";
|
|
}
|
|
|
|
lldb::ProcessSP SBProcess::GetSP() const { return m_opaque_wp.lock(); }
|
|
|
|
void SBProcess::SetSP(const ProcessSP &process_sp) { m_opaque_wp = process_sp; }
|
|
|
|
void SBProcess::Clear() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(void, SBProcess, Clear);
|
|
|
|
m_opaque_wp.reset();
|
|
}
|
|
|
|
bool SBProcess::IsValid() const {
|
|
LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBProcess, IsValid);
|
|
return this->operator bool();
|
|
}
|
|
SBProcess::operator bool() const {
|
|
LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBProcess, operator bool);
|
|
|
|
ProcessSP process_sp(m_opaque_wp.lock());
|
|
return ((bool)process_sp && process_sp->IsValid());
|
|
}
|
|
|
|
bool SBProcess::RemoteLaunch(char const **argv, char const **envp,
|
|
const char *stdin_path, const char *stdout_path,
|
|
const char *stderr_path,
|
|
const char *working_directory,
|
|
uint32_t launch_flags, bool stop_at_entry,
|
|
lldb::SBError &error) {
|
|
LLDB_RECORD_METHOD(bool, SBProcess, RemoteLaunch,
|
|
(const char **, const char **, const char *, const char *,
|
|
const char *, const char *, uint32_t, bool,
|
|
lldb::SBError &),
|
|
argv, envp, stdin_path, stdout_path, stderr_path,
|
|
working_directory, launch_flags, stop_at_entry, error);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
if (process_sp->GetState() == eStateConnected) {
|
|
if (stop_at_entry)
|
|
launch_flags |= eLaunchFlagStopAtEntry;
|
|
ProcessLaunchInfo launch_info(FileSpec(stdin_path), FileSpec(stdout_path),
|
|
FileSpec(stderr_path),
|
|
FileSpec(working_directory), launch_flags);
|
|
Module *exe_module = process_sp->GetTarget().GetExecutableModulePointer();
|
|
if (exe_module)
|
|
launch_info.SetExecutableFile(exe_module->GetPlatformFileSpec(), true);
|
|
if (argv)
|
|
launch_info.GetArguments().AppendArguments(argv);
|
|
if (envp)
|
|
launch_info.GetEnvironment() = Environment(envp);
|
|
error.SetError(process_sp->Launch(launch_info));
|
|
} else {
|
|
error.SetErrorString("must be in eStateConnected to call RemoteLaunch");
|
|
}
|
|
} else {
|
|
error.SetErrorString("unable to attach pid");
|
|
}
|
|
|
|
return error.Success();
|
|
}
|
|
|
|
bool SBProcess::RemoteAttachToProcessWithID(lldb::pid_t pid,
|
|
lldb::SBError &error) {
|
|
LLDB_RECORD_METHOD(bool, SBProcess, RemoteAttachToProcessWithID,
|
|
(lldb::pid_t, lldb::SBError &), pid, error);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
if (process_sp->GetState() == eStateConnected) {
|
|
ProcessAttachInfo attach_info;
|
|
attach_info.SetProcessID(pid);
|
|
error.SetError(process_sp->Attach(attach_info));
|
|
} else {
|
|
error.SetErrorString(
|
|
"must be in eStateConnected to call RemoteAttachToProcessWithID");
|
|
}
|
|
} else {
|
|
error.SetErrorString("unable to attach pid");
|
|
}
|
|
|
|
return error.Success();
|
|
}
|
|
|
|
uint32_t SBProcess::GetNumThreads() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBProcess, GetNumThreads);
|
|
|
|
uint32_t num_threads = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
|
|
const bool can_update = stop_locker.TryLock(&process_sp->GetRunLock());
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
num_threads = process_sp->GetThreadList().GetSize(can_update);
|
|
}
|
|
|
|
return num_threads;
|
|
}
|
|
|
|
SBThread SBProcess::GetSelectedThread() const {
|
|
LLDB_RECORD_METHOD_CONST_NO_ARGS(lldb::SBThread, SBProcess,
|
|
GetSelectedThread);
|
|
|
|
SBThread sb_thread;
|
|
ThreadSP thread_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
thread_sp = process_sp->GetThreadList().GetSelectedThread();
|
|
sb_thread.SetThread(thread_sp);
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_thread);
|
|
}
|
|
|
|
SBThread SBProcess::CreateOSPluginThread(lldb::tid_t tid,
|
|
lldb::addr_t context) {
|
|
LLDB_RECORD_METHOD(lldb::SBThread, SBProcess, CreateOSPluginThread,
|
|
(lldb::tid_t, lldb::addr_t), tid, context);
|
|
|
|
SBThread sb_thread;
|
|
ThreadSP thread_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
thread_sp = process_sp->CreateOSPluginThread(tid, context);
|
|
sb_thread.SetThread(thread_sp);
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_thread);
|
|
}
|
|
|
|
SBTarget SBProcess::GetTarget() const {
|
|
LLDB_RECORD_METHOD_CONST_NO_ARGS(lldb::SBTarget, SBProcess, GetTarget);
|
|
|
|
SBTarget sb_target;
|
|
TargetSP target_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
target_sp = process_sp->GetTarget().shared_from_this();
|
|
sb_target.SetSP(target_sp);
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_target);
|
|
}
|
|
|
|
size_t SBProcess::PutSTDIN(const char *src, size_t src_len) {
|
|
LLDB_RECORD_METHOD(size_t, SBProcess, PutSTDIN, (const char *, size_t), src,
|
|
src_len);
|
|
|
|
size_t ret_val = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Status error;
|
|
ret_val = process_sp->PutSTDIN(src, src_len, error);
|
|
}
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
size_t SBProcess::GetSTDOUT(char *dst, size_t dst_len) const {
|
|
LLDB_RECORD_CHAR_PTR_METHOD_CONST(size_t, SBProcess, GetSTDOUT,
|
|
(char *, size_t), dst, "", dst_len);
|
|
|
|
size_t bytes_read = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Status error;
|
|
bytes_read = process_sp->GetSTDOUT(dst, dst_len, error);
|
|
}
|
|
|
|
return bytes_read;
|
|
}
|
|
|
|
size_t SBProcess::GetSTDERR(char *dst, size_t dst_len) const {
|
|
LLDB_RECORD_CHAR_PTR_METHOD_CONST(size_t, SBProcess, GetSTDERR,
|
|
(char *, size_t), dst, "", dst_len);
|
|
|
|
size_t bytes_read = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Status error;
|
|
bytes_read = process_sp->GetSTDERR(dst, dst_len, error);
|
|
}
|
|
|
|
return bytes_read;
|
|
}
|
|
|
|
size_t SBProcess::GetAsyncProfileData(char *dst, size_t dst_len) const {
|
|
LLDB_RECORD_CHAR_PTR_METHOD_CONST(size_t, SBProcess, GetAsyncProfileData,
|
|
(char *, size_t), dst, "", dst_len);
|
|
|
|
size_t bytes_read = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Status error;
|
|
bytes_read = process_sp->GetAsyncProfileData(dst, dst_len, error);
|
|
}
|
|
|
|
return bytes_read;
|
|
}
|
|
|
|
void SBProcess::ReportEventState(const SBEvent &event, SBFile out) const {
|
|
LLDB_RECORD_METHOD_CONST(void, SBProcess, ReportEventState,
|
|
(const SBEvent &, SBFile), event, out);
|
|
|
|
return ReportEventState(event, out.m_opaque_sp);
|
|
}
|
|
|
|
void SBProcess::ReportEventState(const SBEvent &event, FILE *out) const {
|
|
LLDB_RECORD_METHOD_CONST(void, SBProcess, ReportEventState,
|
|
(const lldb::SBEvent &, FILE *), event, out);
|
|
FileSP outfile = std::make_shared<NativeFile>(out, false);
|
|
return ReportEventState(event, outfile);
|
|
}
|
|
|
|
void SBProcess::ReportEventState(const SBEvent &event, FileSP out) const {
|
|
|
|
LLDB_RECORD_METHOD_CONST(void, SBProcess, ReportEventState,
|
|
(const SBEvent &, FileSP), event, out);
|
|
|
|
if (!out || !out->IsValid())
|
|
return;
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
StreamFile stream(out);
|
|
const StateType event_state = SBProcess::GetStateFromEvent(event);
|
|
stream.Printf("Process %" PRIu64 " %s\n",
|
|
process_sp->GetID(), SBDebugger::StateAsCString(event_state));
|
|
}
|
|
}
|
|
|
|
void SBProcess::AppendEventStateReport(const SBEvent &event,
|
|
SBCommandReturnObject &result) {
|
|
LLDB_RECORD_METHOD(void, SBProcess, AppendEventStateReport,
|
|
(const lldb::SBEvent &, lldb::SBCommandReturnObject &),
|
|
event, result);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
const StateType event_state = SBProcess::GetStateFromEvent(event);
|
|
char message[1024];
|
|
::snprintf(message, sizeof(message), "Process %" PRIu64 " %s\n",
|
|
process_sp->GetID(), SBDebugger::StateAsCString(event_state));
|
|
|
|
result.AppendMessage(message);
|
|
}
|
|
}
|
|
|
|
bool SBProcess::SetSelectedThread(const SBThread &thread) {
|
|
LLDB_RECORD_METHOD(bool, SBProcess, SetSelectedThread,
|
|
(const lldb::SBThread &), thread);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
return process_sp->GetThreadList().SetSelectedThreadByID(
|
|
thread.GetThreadID());
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool SBProcess::SetSelectedThreadByID(lldb::tid_t tid) {
|
|
LLDB_RECORD_METHOD(bool, SBProcess, SetSelectedThreadByID, (lldb::tid_t),
|
|
tid);
|
|
|
|
|
|
bool ret_val = false;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
ret_val = process_sp->GetThreadList().SetSelectedThreadByID(tid);
|
|
}
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
bool SBProcess::SetSelectedThreadByIndexID(uint32_t index_id) {
|
|
LLDB_RECORD_METHOD(bool, SBProcess, SetSelectedThreadByIndexID, (uint32_t),
|
|
index_id);
|
|
|
|
bool ret_val = false;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
ret_val = process_sp->GetThreadList().SetSelectedThreadByIndexID(index_id);
|
|
}
|
|
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
SBThread SBProcess::GetThreadAtIndex(size_t index) {
|
|
LLDB_RECORD_METHOD(lldb::SBThread, SBProcess, GetThreadAtIndex, (size_t),
|
|
index);
|
|
|
|
SBThread sb_thread;
|
|
ThreadSP thread_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
const bool can_update = stop_locker.TryLock(&process_sp->GetRunLock());
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
thread_sp = process_sp->GetThreadList().GetThreadAtIndex(index, can_update);
|
|
sb_thread.SetThread(thread_sp);
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_thread);
|
|
}
|
|
|
|
uint32_t SBProcess::GetNumQueues() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBProcess, GetNumQueues);
|
|
|
|
uint32_t num_queues = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
num_queues = process_sp->GetQueueList().GetSize();
|
|
}
|
|
}
|
|
|
|
return num_queues;
|
|
}
|
|
|
|
SBQueue SBProcess::GetQueueAtIndex(size_t index) {
|
|
LLDB_RECORD_METHOD(lldb::SBQueue, SBProcess, GetQueueAtIndex, (size_t),
|
|
index);
|
|
|
|
SBQueue sb_queue;
|
|
QueueSP queue_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
queue_sp = process_sp->GetQueueList().GetQueueAtIndex(index);
|
|
sb_queue.SetQueue(queue_sp);
|
|
}
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_queue);
|
|
}
|
|
|
|
uint32_t SBProcess::GetStopID(bool include_expression_stops) {
|
|
LLDB_RECORD_METHOD(uint32_t, SBProcess, GetStopID, (bool),
|
|
include_expression_stops);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
if (include_expression_stops)
|
|
return process_sp->GetStopID();
|
|
else
|
|
return process_sp->GetLastNaturalStopID();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
SBEvent SBProcess::GetStopEventForStopID(uint32_t stop_id) {
|
|
LLDB_RECORD_METHOD(lldb::SBEvent, SBProcess, GetStopEventForStopID,
|
|
(uint32_t), stop_id);
|
|
|
|
SBEvent sb_event;
|
|
EventSP event_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
event_sp = process_sp->GetStopEventForStopID(stop_id);
|
|
sb_event.reset(event_sp);
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_event);
|
|
}
|
|
|
|
StateType SBProcess::GetState() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::StateType, SBProcess, GetState);
|
|
|
|
StateType ret_val = eStateInvalid;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
ret_val = process_sp->GetState();
|
|
}
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
int SBProcess::GetExitStatus() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(int, SBProcess, GetExitStatus);
|
|
|
|
int exit_status = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
exit_status = process_sp->GetExitStatus();
|
|
}
|
|
|
|
return exit_status;
|
|
}
|
|
|
|
const char *SBProcess::GetExitDescription() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(const char *, SBProcess, GetExitDescription);
|
|
|
|
const char *exit_desc = nullptr;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
exit_desc = process_sp->GetExitDescription();
|
|
}
|
|
return exit_desc;
|
|
}
|
|
|
|
lldb::pid_t SBProcess::GetProcessID() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::pid_t, SBProcess, GetProcessID);
|
|
|
|
lldb::pid_t ret_val = LLDB_INVALID_PROCESS_ID;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp)
|
|
ret_val = process_sp->GetID();
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
uint32_t SBProcess::GetUniqueID() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBProcess, GetUniqueID);
|
|
|
|
uint32_t ret_val = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp)
|
|
ret_val = process_sp->GetUniqueID();
|
|
return ret_val;
|
|
}
|
|
|
|
ByteOrder SBProcess::GetByteOrder() const {
|
|
LLDB_RECORD_METHOD_CONST_NO_ARGS(lldb::ByteOrder, SBProcess, GetByteOrder);
|
|
|
|
ByteOrder byteOrder = eByteOrderInvalid;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp)
|
|
byteOrder = process_sp->GetTarget().GetArchitecture().GetByteOrder();
|
|
|
|
|
|
return byteOrder;
|
|
}
|
|
|
|
uint32_t SBProcess::GetAddressByteSize() const {
|
|
LLDB_RECORD_METHOD_CONST_NO_ARGS(uint32_t, SBProcess, GetAddressByteSize);
|
|
|
|
uint32_t size = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp)
|
|
size = process_sp->GetTarget().GetArchitecture().GetAddressByteSize();
|
|
|
|
|
|
return size;
|
|
}
|
|
|
|
SBError SBProcess::Continue() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBError, SBProcess, Continue);
|
|
|
|
SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
|
|
if (process_sp->GetTarget().GetDebugger().GetAsyncExecution())
|
|
sb_error.ref() = process_sp->Resume();
|
|
else
|
|
sb_error.ref() = process_sp->ResumeSynchronous(nullptr);
|
|
} else
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
SBError SBProcess::Destroy() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBError, SBProcess, Destroy);
|
|
|
|
SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
sb_error.SetError(process_sp->Destroy(false));
|
|
} else
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
SBError SBProcess::Stop() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBError, SBProcess, Stop);
|
|
|
|
SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
sb_error.SetError(process_sp->Halt());
|
|
} else
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
SBError SBProcess::Kill() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBError, SBProcess, Kill);
|
|
|
|
SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
sb_error.SetError(process_sp->Destroy(true));
|
|
} else
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
SBError SBProcess::Detach() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBError, SBProcess, Detach);
|
|
|
|
// FIXME: This should come from a process default.
|
|
bool keep_stopped = false;
|
|
return LLDB_RECORD_RESULT(Detach(keep_stopped));
|
|
}
|
|
|
|
SBError SBProcess::Detach(bool keep_stopped) {
|
|
LLDB_RECORD_METHOD(lldb::SBError, SBProcess, Detach, (bool), keep_stopped);
|
|
|
|
SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
sb_error.SetError(process_sp->Detach(keep_stopped));
|
|
} else
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
SBError SBProcess::Signal(int signo) {
|
|
LLDB_RECORD_METHOD(lldb::SBError, SBProcess, Signal, (int), signo);
|
|
|
|
SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
sb_error.SetError(process_sp->Signal(signo));
|
|
} else
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
SBUnixSignals SBProcess::GetUnixSignals() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBUnixSignals, SBProcess, GetUnixSignals);
|
|
|
|
if (auto process_sp = GetSP())
|
|
return LLDB_RECORD_RESULT(SBUnixSignals{process_sp});
|
|
|
|
return LLDB_RECORD_RESULT(SBUnixSignals{});
|
|
}
|
|
|
|
void SBProcess::SendAsyncInterrupt() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(void, SBProcess, SendAsyncInterrupt);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
process_sp->SendAsyncInterrupt();
|
|
}
|
|
}
|
|
|
|
SBThread SBProcess::GetThreadByID(tid_t tid) {
|
|
LLDB_RECORD_METHOD(lldb::SBThread, SBProcess, GetThreadByID, (lldb::tid_t),
|
|
tid);
|
|
|
|
SBThread sb_thread;
|
|
ThreadSP thread_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
const bool can_update = stop_locker.TryLock(&process_sp->GetRunLock());
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
thread_sp = process_sp->GetThreadList().FindThreadByID(tid, can_update);
|
|
sb_thread.SetThread(thread_sp);
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_thread);
|
|
}
|
|
|
|
SBThread SBProcess::GetThreadByIndexID(uint32_t index_id) {
|
|
LLDB_RECORD_METHOD(lldb::SBThread, SBProcess, GetThreadByIndexID, (uint32_t),
|
|
index_id);
|
|
|
|
SBThread sb_thread;
|
|
ThreadSP thread_sp;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
const bool can_update = stop_locker.TryLock(&process_sp->GetRunLock());
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
thread_sp =
|
|
process_sp->GetThreadList().FindThreadByIndexID(index_id, can_update);
|
|
sb_thread.SetThread(thread_sp);
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_thread);
|
|
}
|
|
|
|
StateType SBProcess::GetStateFromEvent(const SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(lldb::StateType, SBProcess, GetStateFromEvent,
|
|
(const lldb::SBEvent &), event);
|
|
|
|
StateType ret_val = Process::ProcessEventData::GetStateFromEvent(event.get());
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
bool SBProcess::GetRestartedFromEvent(const SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(bool, SBProcess, GetRestartedFromEvent,
|
|
(const lldb::SBEvent &), event);
|
|
|
|
bool ret_val = Process::ProcessEventData::GetRestartedFromEvent(event.get());
|
|
|
|
return ret_val;
|
|
}
|
|
|
|
size_t SBProcess::GetNumRestartedReasonsFromEvent(const lldb::SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(size_t, SBProcess, GetNumRestartedReasonsFromEvent,
|
|
(const lldb::SBEvent &), event);
|
|
|
|
return Process::ProcessEventData::GetNumRestartedReasons(event.get());
|
|
}
|
|
|
|
const char *
|
|
SBProcess::GetRestartedReasonAtIndexFromEvent(const lldb::SBEvent &event,
|
|
size_t idx) {
|
|
LLDB_RECORD_STATIC_METHOD(const char *, SBProcess,
|
|
GetRestartedReasonAtIndexFromEvent,
|
|
(const lldb::SBEvent &, size_t), event, idx);
|
|
|
|
return Process::ProcessEventData::GetRestartedReasonAtIndex(event.get(), idx);
|
|
}
|
|
|
|
SBProcess SBProcess::GetProcessFromEvent(const SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(lldb::SBProcess, SBProcess, GetProcessFromEvent,
|
|
(const lldb::SBEvent &), event);
|
|
|
|
ProcessSP process_sp =
|
|
Process::ProcessEventData::GetProcessFromEvent(event.get());
|
|
if (!process_sp) {
|
|
// StructuredData events also know the process they come from. Try that.
|
|
process_sp = EventDataStructuredData::GetProcessFromEvent(event.get());
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(SBProcess(process_sp));
|
|
}
|
|
|
|
bool SBProcess::GetInterruptedFromEvent(const SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(bool, SBProcess, GetInterruptedFromEvent,
|
|
(const lldb::SBEvent &), event);
|
|
|
|
return Process::ProcessEventData::GetInterruptedFromEvent(event.get());
|
|
}
|
|
|
|
lldb::SBStructuredData
|
|
SBProcess::GetStructuredDataFromEvent(const lldb::SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(lldb::SBStructuredData, SBProcess,
|
|
GetStructuredDataFromEvent, (const lldb::SBEvent &),
|
|
event);
|
|
|
|
return LLDB_RECORD_RESULT(SBStructuredData(event.GetSP()));
|
|
}
|
|
|
|
bool SBProcess::EventIsProcessEvent(const SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(bool, SBProcess, EventIsProcessEvent,
|
|
(const lldb::SBEvent &), event);
|
|
|
|
return (event.GetBroadcasterClass() == SBProcess::GetBroadcasterClass()) &&
|
|
!EventIsStructuredDataEvent(event);
|
|
}
|
|
|
|
bool SBProcess::EventIsStructuredDataEvent(const lldb::SBEvent &event) {
|
|
LLDB_RECORD_STATIC_METHOD(bool, SBProcess, EventIsStructuredDataEvent,
|
|
(const lldb::SBEvent &), event);
|
|
|
|
EventSP event_sp = event.GetSP();
|
|
EventData *event_data = event_sp ? event_sp->GetData() : nullptr;
|
|
return event_data && (event_data->GetFlavor() ==
|
|
EventDataStructuredData::GetFlavorString());
|
|
}
|
|
|
|
SBBroadcaster SBProcess::GetBroadcaster() const {
|
|
LLDB_RECORD_METHOD_CONST_NO_ARGS(lldb::SBBroadcaster, SBProcess,
|
|
GetBroadcaster);
|
|
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
|
|
SBBroadcaster broadcaster(process_sp.get(), false);
|
|
|
|
|
|
return LLDB_RECORD_RESULT(broadcaster);
|
|
}
|
|
|
|
const char *SBProcess::GetBroadcasterClass() {
|
|
LLDB_RECORD_STATIC_METHOD_NO_ARGS(const char *, SBProcess,
|
|
GetBroadcasterClass);
|
|
|
|
return Process::GetStaticBroadcasterClass().AsCString();
|
|
}
|
|
|
|
size_t SBProcess::ReadMemory(addr_t addr, void *dst, size_t dst_len,
|
|
SBError &sb_error) {
|
|
LLDB_RECORD_DUMMY(size_t, SBProcess, ReadMemory,
|
|
(lldb::addr_t, void *, size_t, lldb::SBError &), addr, dst,
|
|
dst_len, sb_error);
|
|
|
|
size_t bytes_read = 0;
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
|
|
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
bytes_read = process_sp->ReadMemory(addr, dst, dst_len, sb_error.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
|
|
return bytes_read;
|
|
}
|
|
|
|
size_t SBProcess::ReadCStringFromMemory(addr_t addr, void *buf, size_t size,
|
|
lldb::SBError &sb_error) {
|
|
LLDB_RECORD_DUMMY(size_t, SBProcess, ReadCStringFromMemory,
|
|
(lldb::addr_t, void *, size_t, lldb::SBError &), addr, buf,
|
|
size, sb_error);
|
|
|
|
size_t bytes_read = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
bytes_read = process_sp->ReadCStringFromMemory(addr, (char *)buf, size,
|
|
sb_error.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
return bytes_read;
|
|
}
|
|
|
|
uint64_t SBProcess::ReadUnsignedFromMemory(addr_t addr, uint32_t byte_size,
|
|
lldb::SBError &sb_error) {
|
|
LLDB_RECORD_METHOD(uint64_t, SBProcess, ReadUnsignedFromMemory,
|
|
(lldb::addr_t, uint32_t, lldb::SBError &), addr, byte_size,
|
|
sb_error);
|
|
|
|
uint64_t value = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
value = process_sp->ReadUnsignedIntegerFromMemory(addr, byte_size, 0,
|
|
sb_error.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
return value;
|
|
}
|
|
|
|
lldb::addr_t SBProcess::ReadPointerFromMemory(addr_t addr,
|
|
lldb::SBError &sb_error) {
|
|
LLDB_RECORD_METHOD(lldb::addr_t, SBProcess, ReadPointerFromMemory,
|
|
(lldb::addr_t, lldb::SBError &), addr, sb_error);
|
|
|
|
lldb::addr_t ptr = LLDB_INVALID_ADDRESS;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
ptr = process_sp->ReadPointerFromMemory(addr, sb_error.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
return ptr;
|
|
}
|
|
|
|
size_t SBProcess::WriteMemory(addr_t addr, const void *src, size_t src_len,
|
|
SBError &sb_error) {
|
|
LLDB_RECORD_DUMMY(size_t, SBProcess, WriteMemory,
|
|
(lldb::addr_t, const void *, size_t, lldb::SBError &), addr,
|
|
src, src_len, sb_error);
|
|
|
|
size_t bytes_written = 0;
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
bytes_written =
|
|
process_sp->WriteMemory(addr, src, src_len, sb_error.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
}
|
|
|
|
return bytes_written;
|
|
}
|
|
|
|
bool SBProcess::GetDescription(SBStream &description) {
|
|
LLDB_RECORD_METHOD(bool, SBProcess, GetDescription, (lldb::SBStream &),
|
|
description);
|
|
|
|
Stream &strm = description.ref();
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
char path[PATH_MAX];
|
|
GetTarget().GetExecutable().GetPath(path, sizeof(path));
|
|
Module *exe_module = process_sp->GetTarget().GetExecutableModulePointer();
|
|
const char *exe_name = nullptr;
|
|
if (exe_module)
|
|
exe_name = exe_module->GetFileSpec().GetFilename().AsCString();
|
|
|
|
strm.Printf("SBProcess: pid = %" PRIu64 ", state = %s, threads = %d%s%s",
|
|
process_sp->GetID(), lldb_private::StateAsCString(GetState()),
|
|
GetNumThreads(), exe_name ? ", executable = " : "",
|
|
exe_name ? exe_name : "");
|
|
} else
|
|
strm.PutCString("No value");
|
|
|
|
return true;
|
|
}
|
|
|
|
SBStructuredData SBProcess::GetExtendedCrashInformation() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBStructuredData, SBProcess,
|
|
GetExtendedCrashInformation);
|
|
SBStructuredData data;
|
|
ProcessSP process_sp(GetSP());
|
|
if (!process_sp)
|
|
return LLDB_RECORD_RESULT(data);
|
|
|
|
PlatformSP platform_sp = process_sp->GetTarget().GetPlatform();
|
|
|
|
if (!platform_sp)
|
|
return LLDB_RECORD_RESULT(data);
|
|
|
|
auto expected_data =
|
|
platform_sp->FetchExtendedCrashInformation(*process_sp.get());
|
|
|
|
if (!expected_data)
|
|
return LLDB_RECORD_RESULT(data);
|
|
|
|
StructuredData::ObjectSP fetched_data = *expected_data;
|
|
data.m_impl_up->SetObjectSP(fetched_data);
|
|
return LLDB_RECORD_RESULT(data);
|
|
}
|
|
|
|
uint32_t
|
|
SBProcess::GetNumSupportedHardwareWatchpoints(lldb::SBError &sb_error) const {
|
|
LLDB_RECORD_METHOD_CONST(uint32_t, SBProcess,
|
|
GetNumSupportedHardwareWatchpoints,
|
|
(lldb::SBError &), sb_error);
|
|
|
|
uint32_t num = 0;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
sb_error.SetError(process_sp->GetWatchpointSupportInfo(num));
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
return num;
|
|
}
|
|
|
|
uint32_t SBProcess::LoadImage(lldb::SBFileSpec &sb_remote_image_spec,
|
|
lldb::SBError &sb_error) {
|
|
LLDB_RECORD_METHOD(uint32_t, SBProcess, LoadImage,
|
|
(lldb::SBFileSpec &, lldb::SBError &),
|
|
sb_remote_image_spec, sb_error);
|
|
|
|
return LoadImage(SBFileSpec(), sb_remote_image_spec, sb_error);
|
|
}
|
|
|
|
uint32_t SBProcess::LoadImage(const lldb::SBFileSpec &sb_local_image_spec,
|
|
const lldb::SBFileSpec &sb_remote_image_spec,
|
|
lldb::SBError &sb_error) {
|
|
LLDB_RECORD_METHOD(
|
|
uint32_t, SBProcess, LoadImage,
|
|
(const lldb::SBFileSpec &, const lldb::SBFileSpec &, lldb::SBError &),
|
|
sb_local_image_spec, sb_remote_image_spec, sb_error);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
PlatformSP platform_sp = process_sp->GetTarget().GetPlatform();
|
|
return platform_sp->LoadImage(process_sp.get(), *sb_local_image_spec,
|
|
*sb_remote_image_spec, sb_error.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("process is invalid");
|
|
}
|
|
return LLDB_INVALID_IMAGE_TOKEN;
|
|
}
|
|
|
|
uint32_t SBProcess::LoadImageUsingPaths(const lldb::SBFileSpec &image_spec,
|
|
SBStringList &paths,
|
|
lldb::SBFileSpec &loaded_path,
|
|
lldb::SBError &error) {
|
|
LLDB_RECORD_METHOD(uint32_t, SBProcess, LoadImageUsingPaths,
|
|
(const lldb::SBFileSpec &, lldb::SBStringList &,
|
|
lldb::SBFileSpec &, lldb::SBError &),
|
|
image_spec, paths, loaded_path, error);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
PlatformSP platform_sp = process_sp->GetTarget().GetPlatform();
|
|
size_t num_paths = paths.GetSize();
|
|
std::vector<std::string> paths_vec;
|
|
paths_vec.reserve(num_paths);
|
|
for (size_t i = 0; i < num_paths; i++)
|
|
paths_vec.push_back(paths.GetStringAtIndex(i));
|
|
FileSpec loaded_spec;
|
|
|
|
uint32_t token = platform_sp->LoadImageUsingPaths(
|
|
process_sp.get(), *image_spec, paths_vec, error.ref(), &loaded_spec);
|
|
if (token != LLDB_INVALID_IMAGE_TOKEN)
|
|
loaded_path = loaded_spec;
|
|
return token;
|
|
} else {
|
|
error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
error.SetErrorString("process is invalid");
|
|
}
|
|
|
|
return LLDB_INVALID_IMAGE_TOKEN;
|
|
}
|
|
|
|
lldb::SBError SBProcess::UnloadImage(uint32_t image_token) {
|
|
LLDB_RECORD_METHOD(lldb::SBError, SBProcess, UnloadImage, (uint32_t),
|
|
image_token);
|
|
|
|
lldb::SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
PlatformSP platform_sp = process_sp->GetTarget().GetPlatform();
|
|
sb_error.SetError(
|
|
platform_sp->UnloadImage(process_sp.get(), image_token));
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else
|
|
sb_error.SetErrorString("invalid process");
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
lldb::SBError SBProcess::SendEventData(const char *event_data) {
|
|
LLDB_RECORD_METHOD(lldb::SBError, SBProcess, SendEventData, (const char *),
|
|
event_data);
|
|
|
|
lldb::SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
sb_error.SetError(process_sp->SendEventData(event_data));
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else
|
|
sb_error.SetErrorString("invalid process");
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
uint32_t SBProcess::GetNumExtendedBacktraceTypes() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBProcess, GetNumExtendedBacktraceTypes);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp && process_sp->GetSystemRuntime()) {
|
|
SystemRuntime *runtime = process_sp->GetSystemRuntime();
|
|
return runtime->GetExtendedBacktraceTypes().size();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
const char *SBProcess::GetExtendedBacktraceTypeAtIndex(uint32_t idx) {
|
|
LLDB_RECORD_METHOD(const char *, SBProcess, GetExtendedBacktraceTypeAtIndex,
|
|
(uint32_t), idx);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp && process_sp->GetSystemRuntime()) {
|
|
SystemRuntime *runtime = process_sp->GetSystemRuntime();
|
|
const std::vector<ConstString> &names =
|
|
runtime->GetExtendedBacktraceTypes();
|
|
if (idx < names.size()) {
|
|
return names[idx].AsCString();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
SBThreadCollection SBProcess::GetHistoryThreads(addr_t addr) {
|
|
LLDB_RECORD_METHOD(lldb::SBThreadCollection, SBProcess, GetHistoryThreads,
|
|
(lldb::addr_t), addr);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
SBThreadCollection threads;
|
|
if (process_sp) {
|
|
threads = SBThreadCollection(process_sp->GetHistoryThreads(addr));
|
|
}
|
|
return LLDB_RECORD_RESULT(threads);
|
|
}
|
|
|
|
bool SBProcess::IsInstrumentationRuntimePresent(
|
|
InstrumentationRuntimeType type) {
|
|
LLDB_RECORD_METHOD(bool, SBProcess, IsInstrumentationRuntimePresent,
|
|
(lldb::InstrumentationRuntimeType), type);
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
if (!process_sp)
|
|
return false;
|
|
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
|
|
InstrumentationRuntimeSP runtime_sp =
|
|
process_sp->GetInstrumentationRuntime(type);
|
|
|
|
if (!runtime_sp.get())
|
|
return false;
|
|
|
|
return runtime_sp->IsActive();
|
|
}
|
|
|
|
lldb::SBError SBProcess::SaveCore(const char *file_name) {
|
|
LLDB_RECORD_METHOD(lldb::SBError, SBProcess, SaveCore, (const char *),
|
|
file_name);
|
|
|
|
lldb::SBError error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (!process_sp) {
|
|
error.SetErrorString("SBProcess is invalid");
|
|
return LLDB_RECORD_RESULT(error);
|
|
}
|
|
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
|
|
if (process_sp->GetState() != eStateStopped) {
|
|
error.SetErrorString("the process is not stopped");
|
|
return LLDB_RECORD_RESULT(error);
|
|
}
|
|
|
|
FileSpec core_file(file_name);
|
|
SaveCoreStyle core_style = SaveCoreStyle::eSaveCoreFull;
|
|
error.ref() = PluginManager::SaveCore(process_sp, core_file, core_style);
|
|
return LLDB_RECORD_RESULT(error);
|
|
}
|
|
|
|
lldb::SBError
|
|
SBProcess::GetMemoryRegionInfo(lldb::addr_t load_addr,
|
|
SBMemoryRegionInfo &sb_region_info) {
|
|
LLDB_RECORD_METHOD(lldb::SBError, SBProcess, GetMemoryRegionInfo,
|
|
(lldb::addr_t, lldb::SBMemoryRegionInfo &), load_addr,
|
|
sb_region_info);
|
|
|
|
lldb::SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
|
|
sb_error.ref() =
|
|
process_sp->GetMemoryRegionInfo(load_addr, sb_region_info.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
return LLDB_RECORD_RESULT(sb_error);
|
|
}
|
|
|
|
lldb::SBMemoryRegionInfoList SBProcess::GetMemoryRegions() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBMemoryRegionInfoList, SBProcess,
|
|
GetMemoryRegions);
|
|
|
|
lldb::SBMemoryRegionInfoList sb_region_list;
|
|
|
|
ProcessSP process_sp(GetSP());
|
|
Process::StopLocker stop_locker;
|
|
if (process_sp && stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
|
|
process_sp->GetMemoryRegions(sb_region_list.ref());
|
|
}
|
|
|
|
return LLDB_RECORD_RESULT(sb_region_list);
|
|
}
|
|
|
|
lldb::SBProcessInfo SBProcess::GetProcessInfo() {
|
|
LLDB_RECORD_METHOD_NO_ARGS(lldb::SBProcessInfo, SBProcess, GetProcessInfo);
|
|
|
|
lldb::SBProcessInfo sb_proc_info;
|
|
ProcessSP process_sp(GetSP());
|
|
ProcessInstanceInfo proc_info;
|
|
if (process_sp && process_sp->GetProcessInfo(proc_info)) {
|
|
sb_proc_info.SetProcessInfo(proc_info);
|
|
}
|
|
return LLDB_RECORD_RESULT(sb_proc_info);
|
|
}
|
|
|
|
lldb::addr_t SBProcess::AllocateMemory(size_t size, uint32_t permissions,
|
|
lldb::SBError &sb_error) {
|
|
LLDB_RECORD_METHOD(lldb::addr_t, SBProcess, AllocateMemory,
|
|
(size_t, uint32_t, lldb::SBError &), size, permissions,
|
|
sb_error);
|
|
|
|
lldb::addr_t addr = LLDB_INVALID_ADDRESS;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
addr = process_sp->AllocateMemory(size, permissions, sb_error.ref());
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
return addr;
|
|
}
|
|
|
|
lldb::SBError SBProcess::DeallocateMemory(lldb::addr_t ptr) {
|
|
LLDB_RECORD_METHOD(lldb::SBError, SBProcess, DeallocateMemory, (lldb::addr_t),
|
|
ptr);
|
|
|
|
lldb::SBError sb_error;
|
|
ProcessSP process_sp(GetSP());
|
|
if (process_sp) {
|
|
Process::StopLocker stop_locker;
|
|
if (stop_locker.TryLock(&process_sp->GetRunLock())) {
|
|
std::lock_guard<std::recursive_mutex> guard(
|
|
process_sp->GetTarget().GetAPIMutex());
|
|
Status error = process_sp->DeallocateMemory(ptr);
|
|
sb_error.SetError(error);
|
|
} else {
|
|
sb_error.SetErrorString("process is running");
|
|
}
|
|
} else {
|
|
sb_error.SetErrorString("SBProcess is invalid");
|
|
}
|
|
return sb_error;
|
|
}
|
|
|
|
namespace lldb_private {
|
|
namespace repro {
|
|
|
|
template <>
|
|
void RegisterMethods<SBProcess>(Registry &R) {
|
|
LLDB_REGISTER_CONSTRUCTOR(SBProcess, ());
|
|
LLDB_REGISTER_CONSTRUCTOR(SBProcess, (const lldb::SBProcess &));
|
|
LLDB_REGISTER_CONSTRUCTOR(SBProcess, (const lldb::ProcessSP &));
|
|
LLDB_REGISTER_METHOD(const lldb::SBProcess &,
|
|
SBProcess, operator=,(const lldb::SBProcess &));
|
|
LLDB_REGISTER_STATIC_METHOD(const char *, SBProcess,
|
|
GetBroadcasterClassName, ());
|
|
LLDB_REGISTER_METHOD(const char *, SBProcess, GetPluginName, ());
|
|
LLDB_REGISTER_METHOD(const char *, SBProcess, GetShortPluginName, ());
|
|
LLDB_REGISTER_METHOD(void, SBProcess, Clear, ());
|
|
LLDB_REGISTER_METHOD_CONST(bool, SBProcess, IsValid, ());
|
|
LLDB_REGISTER_METHOD_CONST(bool, SBProcess, operator bool, ());
|
|
LLDB_REGISTER_METHOD(bool, SBProcess, RemoteLaunch,
|
|
(const char **, const char **, const char *,
|
|
const char *, const char *, const char *, uint32_t,
|
|
bool, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(bool, SBProcess, RemoteAttachToProcessWithID,
|
|
(lldb::pid_t, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(uint32_t, SBProcess, GetNumThreads, ());
|
|
LLDB_REGISTER_METHOD_CONST(lldb::SBThread, SBProcess, GetSelectedThread,
|
|
());
|
|
LLDB_REGISTER_METHOD(lldb::SBThread, SBProcess, CreateOSPluginThread,
|
|
(lldb::tid_t, lldb::addr_t));
|
|
LLDB_REGISTER_METHOD_CONST(lldb::SBTarget, SBProcess, GetTarget, ());
|
|
LLDB_REGISTER_METHOD(size_t, SBProcess, PutSTDIN, (const char *, size_t));
|
|
LLDB_REGISTER_METHOD_CONST(void, SBProcess, ReportEventState,
|
|
(const lldb::SBEvent &, FILE *));
|
|
LLDB_REGISTER_METHOD_CONST(void, SBProcess, ReportEventState,
|
|
(const lldb::SBEvent &, FileSP));
|
|
LLDB_REGISTER_METHOD_CONST(void, SBProcess, ReportEventState,
|
|
(const lldb::SBEvent &, SBFile));
|
|
LLDB_REGISTER_METHOD(
|
|
void, SBProcess, AppendEventStateReport,
|
|
(const lldb::SBEvent &, lldb::SBCommandReturnObject &));
|
|
LLDB_REGISTER_METHOD(bool, SBProcess, SetSelectedThread,
|
|
(const lldb::SBThread &));
|
|
LLDB_REGISTER_METHOD(bool, SBProcess, SetSelectedThreadByID, (lldb::tid_t));
|
|
LLDB_REGISTER_METHOD(bool, SBProcess, SetSelectedThreadByIndexID,
|
|
(uint32_t));
|
|
LLDB_REGISTER_METHOD(lldb::SBThread, SBProcess, GetThreadAtIndex, (size_t));
|
|
LLDB_REGISTER_METHOD(uint32_t, SBProcess, GetNumQueues, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBQueue, SBProcess, GetQueueAtIndex, (size_t));
|
|
LLDB_REGISTER_METHOD(uint32_t, SBProcess, GetStopID, (bool));
|
|
LLDB_REGISTER_METHOD(lldb::SBEvent, SBProcess, GetStopEventForStopID,
|
|
(uint32_t));
|
|
LLDB_REGISTER_METHOD(lldb::StateType, SBProcess, GetState, ());
|
|
LLDB_REGISTER_METHOD(int, SBProcess, GetExitStatus, ());
|
|
LLDB_REGISTER_METHOD(const char *, SBProcess, GetExitDescription, ());
|
|
LLDB_REGISTER_METHOD(lldb::pid_t, SBProcess, GetProcessID, ());
|
|
LLDB_REGISTER_METHOD(uint32_t, SBProcess, GetUniqueID, ());
|
|
LLDB_REGISTER_METHOD_CONST(lldb::ByteOrder, SBProcess, GetByteOrder, ());
|
|
LLDB_REGISTER_METHOD_CONST(uint32_t, SBProcess, GetAddressByteSize, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, Continue, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, Destroy, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, Stop, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, Kill, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, Detach, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, Detach, (bool));
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, Signal, (int));
|
|
LLDB_REGISTER_METHOD(lldb::SBUnixSignals, SBProcess, GetUnixSignals, ());
|
|
LLDB_REGISTER_METHOD(void, SBProcess, SendAsyncInterrupt, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBThread, SBProcess, GetThreadByID,
|
|
(lldb::tid_t));
|
|
LLDB_REGISTER_METHOD(lldb::SBThread, SBProcess, GetThreadByIndexID,
|
|
(uint32_t));
|
|
LLDB_REGISTER_STATIC_METHOD(lldb::StateType, SBProcess, GetStateFromEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_STATIC_METHOD(bool, SBProcess, GetRestartedFromEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_STATIC_METHOD(size_t, SBProcess,
|
|
GetNumRestartedReasonsFromEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_STATIC_METHOD(const char *, SBProcess,
|
|
GetRestartedReasonAtIndexFromEvent,
|
|
(const lldb::SBEvent &, size_t));
|
|
LLDB_REGISTER_STATIC_METHOD(lldb::SBProcess, SBProcess, GetProcessFromEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_STATIC_METHOD(bool, SBProcess, GetInterruptedFromEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_STATIC_METHOD(lldb::SBStructuredData, SBProcess,
|
|
GetStructuredDataFromEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_STATIC_METHOD(bool, SBProcess, EventIsProcessEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_STATIC_METHOD(bool, SBProcess, EventIsStructuredDataEvent,
|
|
(const lldb::SBEvent &));
|
|
LLDB_REGISTER_METHOD_CONST(lldb::SBBroadcaster, SBProcess, GetBroadcaster,
|
|
());
|
|
LLDB_REGISTER_STATIC_METHOD(const char *, SBProcess, GetBroadcasterClass,
|
|
());
|
|
LLDB_REGISTER_METHOD(uint64_t, SBProcess, ReadUnsignedFromMemory,
|
|
(lldb::addr_t, uint32_t, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(lldb::addr_t, SBProcess, ReadPointerFromMemory,
|
|
(lldb::addr_t, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(bool, SBProcess, GetDescription, (lldb::SBStream &));
|
|
LLDB_REGISTER_METHOD(lldb::SBStructuredData, SBProcess,
|
|
GetExtendedCrashInformation, ());
|
|
LLDB_REGISTER_METHOD_CONST(uint32_t, SBProcess,
|
|
GetNumSupportedHardwareWatchpoints,
|
|
(lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(uint32_t, SBProcess, LoadImage,
|
|
(lldb::SBFileSpec &, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(
|
|
uint32_t, SBProcess, LoadImage,
|
|
(const lldb::SBFileSpec &, const lldb::SBFileSpec &, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(uint32_t, SBProcess, LoadImageUsingPaths,
|
|
(const lldb::SBFileSpec &, lldb::SBStringList &,
|
|
lldb::SBFileSpec &, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, UnloadImage, (uint32_t));
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, SendEventData,
|
|
(const char *));
|
|
LLDB_REGISTER_METHOD(uint32_t, SBProcess, GetNumExtendedBacktraceTypes, ());
|
|
LLDB_REGISTER_METHOD(const char *, SBProcess,
|
|
GetExtendedBacktraceTypeAtIndex, (uint32_t));
|
|
LLDB_REGISTER_METHOD(lldb::SBThreadCollection, SBProcess, GetHistoryThreads,
|
|
(lldb::addr_t));
|
|
LLDB_REGISTER_METHOD(bool, SBProcess, IsInstrumentationRuntimePresent,
|
|
(lldb::InstrumentationRuntimeType));
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, SaveCore, (const char *));
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, GetMemoryRegionInfo,
|
|
(lldb::addr_t, lldb::SBMemoryRegionInfo &));
|
|
LLDB_REGISTER_METHOD(lldb::SBMemoryRegionInfoList, SBProcess,
|
|
GetMemoryRegions, ());
|
|
LLDB_REGISTER_METHOD(lldb::SBProcessInfo, SBProcess, GetProcessInfo, ());
|
|
LLDB_REGISTER_METHOD(lldb::addr_t, SBProcess, AllocateMemory,
|
|
(size_t, uint32_t, lldb::SBError &));
|
|
LLDB_REGISTER_METHOD(lldb::SBError, SBProcess, DeallocateMemory,
|
|
(lldb::addr_t));
|
|
|
|
LLDB_REGISTER_CHAR_PTR_METHOD_CONST(size_t, SBProcess, GetSTDOUT);
|
|
LLDB_REGISTER_CHAR_PTR_METHOD_CONST(size_t, SBProcess, GetSTDERR);
|
|
LLDB_REGISTER_CHAR_PTR_METHOD_CONST(size_t, SBProcess, GetAsyncProfileData);
|
|
}
|
|
|
|
}
|
|
}
|