//===-- SBLaunchInfo.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/SBLaunchInfo.h" #include "lldb/Utility/Instrumentation.h" #include "lldb/API/SBEnvironment.h" #include "lldb/API/SBError.h" #include "lldb/API/SBFileSpec.h" #include "lldb/API/SBListener.h" #include "lldb/API/SBStream.h" #include "lldb/API/SBStructuredData.h" #include "lldb/Core/StructuredDataImpl.h" #include "lldb/Host/ProcessLaunchInfo.h" #include "lldb/Utility/Listener.h" #include "lldb/Utility/ScriptedMetadata.h" using namespace lldb; using namespace lldb_private; class lldb_private::SBLaunchInfoImpl : public ProcessLaunchInfo { public: SBLaunchInfoImpl() : m_envp(GetEnvironment().getEnvp()) {} const char *const *GetEnvp() const { return m_envp; } void RegenerateEnvp() { m_envp = GetEnvironment().getEnvp(); } SBLaunchInfoImpl &operator=(const ProcessLaunchInfo &rhs) { ProcessLaunchInfo::operator=(rhs); RegenerateEnvp(); return *this; } private: Environment::Envp m_envp; }; SBLaunchInfo::SBLaunchInfo(const char **argv) : m_opaque_sp(new SBLaunchInfoImpl()) { LLDB_INSTRUMENT_VA(this, argv); m_opaque_sp->GetFlags().Reset(eLaunchFlagDebug | eLaunchFlagDisableASLR); if (argv && argv[0]) m_opaque_sp->GetArguments().SetArguments(argv); } SBLaunchInfo::SBLaunchInfo(const SBLaunchInfo &rhs) { LLDB_INSTRUMENT_VA(this, rhs); m_opaque_sp = rhs.m_opaque_sp; } SBLaunchInfo &SBLaunchInfo::operator=(const SBLaunchInfo &rhs) { LLDB_INSTRUMENT_VA(this, rhs); m_opaque_sp = rhs.m_opaque_sp; return *this; } SBLaunchInfo::~SBLaunchInfo() = default; const lldb_private::ProcessLaunchInfo &SBLaunchInfo::ref() const { return *m_opaque_sp; } void SBLaunchInfo::set_ref(const ProcessLaunchInfo &info) { *m_opaque_sp = info; } lldb::pid_t SBLaunchInfo::GetProcessID() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetProcessID(); } uint32_t SBLaunchInfo::GetUserID() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetUserID(); } uint32_t SBLaunchInfo::GetGroupID() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetGroupID(); } bool SBLaunchInfo::UserIDIsValid() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->UserIDIsValid(); } bool SBLaunchInfo::GroupIDIsValid() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GroupIDIsValid(); } void SBLaunchInfo::SetUserID(uint32_t uid) { LLDB_INSTRUMENT_VA(this, uid); m_opaque_sp->SetUserID(uid); } void SBLaunchInfo::SetGroupID(uint32_t gid) { LLDB_INSTRUMENT_VA(this, gid); m_opaque_sp->SetGroupID(gid); } SBFileSpec SBLaunchInfo::GetExecutableFile() { LLDB_INSTRUMENT_VA(this); return SBFileSpec(m_opaque_sp->GetExecutableFile()); } void SBLaunchInfo::SetExecutableFile(SBFileSpec exe_file, bool add_as_first_arg) { LLDB_INSTRUMENT_VA(this, exe_file, add_as_first_arg); m_opaque_sp->SetExecutableFile(exe_file.ref(), add_as_first_arg); } SBListener SBLaunchInfo::GetListener() { LLDB_INSTRUMENT_VA(this); return SBListener(m_opaque_sp->GetListener()); } void SBLaunchInfo::SetListener(SBListener &listener) { LLDB_INSTRUMENT_VA(this, listener); m_opaque_sp->SetListener(listener.GetSP()); } uint32_t SBLaunchInfo::GetNumArguments() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetArguments().GetArgumentCount(); } const char *SBLaunchInfo::GetArgumentAtIndex(uint32_t idx) { LLDB_INSTRUMENT_VA(this, idx); return ConstString(m_opaque_sp->GetArguments().GetArgumentAtIndex(idx)) .GetCString(); } void SBLaunchInfo::SetArguments(const char **argv, bool append) { LLDB_INSTRUMENT_VA(this, argv, append); if (append) { if (argv) m_opaque_sp->GetArguments().AppendArguments(argv); } else { if (argv) m_opaque_sp->GetArguments().SetArguments(argv); else m_opaque_sp->GetArguments().Clear(); } } uint32_t SBLaunchInfo::GetNumEnvironmentEntries() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetEnvironment().size(); } const char *SBLaunchInfo::GetEnvironmentEntryAtIndex(uint32_t idx) { LLDB_INSTRUMENT_VA(this, idx); if (idx > GetNumEnvironmentEntries()) return nullptr; return ConstString(m_opaque_sp->GetEnvp()[idx]).GetCString(); } void SBLaunchInfo::SetEnvironmentEntries(const char **envp, bool append) { LLDB_INSTRUMENT_VA(this, envp, append); SetEnvironment(SBEnvironment(Environment(envp)), append); } void SBLaunchInfo::SetEnvironment(const SBEnvironment &env, bool append) { LLDB_INSTRUMENT_VA(this, env, append); Environment &refEnv = env.ref(); if (append) { for (auto &KV : refEnv) m_opaque_sp->GetEnvironment().insert_or_assign(KV.first(), KV.second); } else m_opaque_sp->GetEnvironment() = refEnv; m_opaque_sp->RegenerateEnvp(); } SBEnvironment SBLaunchInfo::GetEnvironment() { LLDB_INSTRUMENT_VA(this); return SBEnvironment(Environment(m_opaque_sp->GetEnvironment())); } void SBLaunchInfo::Clear() { LLDB_INSTRUMENT_VA(this); m_opaque_sp->Clear(); } const char *SBLaunchInfo::GetWorkingDirectory() const { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetWorkingDirectory().GetPathAsConstString().AsCString(); } void SBLaunchInfo::SetWorkingDirectory(const char *working_dir) { LLDB_INSTRUMENT_VA(this, working_dir); m_opaque_sp->SetWorkingDirectory(FileSpec(working_dir)); } uint32_t SBLaunchInfo::GetLaunchFlags() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetFlags().Get(); } void SBLaunchInfo::SetLaunchFlags(uint32_t flags) { LLDB_INSTRUMENT_VA(this, flags); m_opaque_sp->GetFlags().Reset(flags); } const char *SBLaunchInfo::GetProcessPluginName() { LLDB_INSTRUMENT_VA(this); return ConstString(m_opaque_sp->GetProcessPluginName()).GetCString(); } void SBLaunchInfo::SetProcessPluginName(const char *plugin_name) { LLDB_INSTRUMENT_VA(this, plugin_name); return m_opaque_sp->SetProcessPluginName(plugin_name); } const char *SBLaunchInfo::GetShell() { LLDB_INSTRUMENT_VA(this); // Constify this string so that it is saved in the string pool. Otherwise it // would be freed when this function goes out of scope. ConstString shell(m_opaque_sp->GetShell().GetPath().c_str()); return shell.AsCString(); } void SBLaunchInfo::SetShell(const char *path) { LLDB_INSTRUMENT_VA(this, path); m_opaque_sp->SetShell(FileSpec(path)); } bool SBLaunchInfo::GetShellExpandArguments() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetShellExpandArguments(); } void SBLaunchInfo::SetShellExpandArguments(bool expand) { LLDB_INSTRUMENT_VA(this, expand); m_opaque_sp->SetShellExpandArguments(expand); } uint32_t SBLaunchInfo::GetResumeCount() { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetResumeCount(); } void SBLaunchInfo::SetResumeCount(uint32_t c) { LLDB_INSTRUMENT_VA(this, c); m_opaque_sp->SetResumeCount(c); } bool SBLaunchInfo::AddCloseFileAction(int fd) { LLDB_INSTRUMENT_VA(this, fd); return m_opaque_sp->AppendCloseFileAction(fd); } bool SBLaunchInfo::AddDuplicateFileAction(int fd, int dup_fd) { LLDB_INSTRUMENT_VA(this, fd, dup_fd); return m_opaque_sp->AppendDuplicateFileAction(fd, dup_fd); } bool SBLaunchInfo::AddOpenFileAction(int fd, const char *path, bool read, bool write) { LLDB_INSTRUMENT_VA(this, fd, path, read, write); return m_opaque_sp->AppendOpenFileAction(fd, FileSpec(path), read, write); } bool SBLaunchInfo::AddSuppressFileAction(int fd, bool read, bool write) { LLDB_INSTRUMENT_VA(this, fd, read, write); return m_opaque_sp->AppendSuppressFileAction(fd, read, write); } void SBLaunchInfo::SetLaunchEventData(const char *data) { LLDB_INSTRUMENT_VA(this, data); m_opaque_sp->SetLaunchEventData(data); } const char *SBLaunchInfo::GetLaunchEventData() const { LLDB_INSTRUMENT_VA(this); return ConstString(m_opaque_sp->GetLaunchEventData()).GetCString(); } void SBLaunchInfo::SetDetachOnError(bool enable) { LLDB_INSTRUMENT_VA(this, enable); m_opaque_sp->SetDetachOnError(enable); } bool SBLaunchInfo::GetDetachOnError() const { LLDB_INSTRUMENT_VA(this); return m_opaque_sp->GetDetachOnError(); } const char *SBLaunchInfo::GetScriptedProcessClassName() const { LLDB_INSTRUMENT_VA(this); ScriptedMetadataSP metadata_sp = m_opaque_sp->GetScriptedMetadata(); if (!metadata_sp || !*metadata_sp) return nullptr; // Constify this string so that it is saved in the string pool. Otherwise it // would be freed when this function goes out of scope. ConstString class_name(metadata_sp->GetClassName().data()); return class_name.AsCString(); } void SBLaunchInfo::SetScriptedProcessClassName(const char *class_name) { LLDB_INSTRUMENT_VA(this, class_name); ScriptedMetadataSP metadata_sp = m_opaque_sp->GetScriptedMetadata(); StructuredData::DictionarySP dict_sp = metadata_sp ? metadata_sp->GetArgsSP() : nullptr; metadata_sp = std::make_shared(class_name, dict_sp); m_opaque_sp->SetScriptedMetadata(metadata_sp); } lldb::SBStructuredData SBLaunchInfo::GetScriptedProcessDictionary() const { LLDB_INSTRUMENT_VA(this); ScriptedMetadataSP metadata_sp = m_opaque_sp->GetScriptedMetadata(); SBStructuredData data; if (!metadata_sp) return data; lldb_private::StructuredData::DictionarySP dict_sp = metadata_sp->GetArgsSP(); data.m_impl_up->SetObjectSP(dict_sp); return data; } void SBLaunchInfo::SetScriptedProcessDictionary(lldb::SBStructuredData dict) { LLDB_INSTRUMENT_VA(this, dict); if (!dict.IsValid() || !dict.m_impl_up) return; StructuredData::ObjectSP obj_sp = dict.m_impl_up->GetObjectSP(); if (!obj_sp) return; StructuredData::DictionarySP dict_sp = std::make_shared(obj_sp); if (!dict_sp || dict_sp->GetType() == lldb::eStructuredDataTypeInvalid) return; ScriptedMetadataSP metadata_sp = m_opaque_sp->GetScriptedMetadata(); llvm::StringRef class_name = metadata_sp ? metadata_sp->GetClassName() : ""; metadata_sp = std::make_shared(class_name, dict_sp); m_opaque_sp->SetScriptedMetadata(metadata_sp); } SBListener SBLaunchInfo::GetShadowListener() { LLDB_INSTRUMENT_VA(this); lldb::ListenerSP shadow_sp = m_opaque_sp->GetShadowListener(); if (!shadow_sp) return SBListener(); return SBListener(shadow_sp); } void SBLaunchInfo::SetShadowListener(SBListener &listener) { LLDB_INSTRUMENT_VA(this, listener); m_opaque_sp->SetShadowListener(listener.GetSP()); }