2020-01-24 08:23:27 +01:00
|
|
|
//===-- HostProcessWindows.cpp --------------------------------------------===//
|
2014-08-27 20:15:30 +00:00
|
|
|
//
|
2019-01-19 08:50:56 +00:00
|
|
|
// 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
|
2014-08-27 20:15:30 +00:00
|
|
|
//
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
|
|
#include "lldb/Host/windows/HostProcessWindows.h"
|
2014-10-14 21:55:08 +00:00
|
|
|
#include "lldb/Host/HostThread.h"
|
|
|
|
|
#include "lldb/Host/ThreadLauncher.h"
|
2014-08-27 20:15:30 +00:00
|
|
|
#include "lldb/Host/windows/windows.h"
|
2017-03-22 18:40:07 +00:00
|
|
|
#include "lldb/Utility/FileSpec.h"
|
2014-08-27 20:15:30 +00:00
|
|
|
|
|
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2016-03-22 17:58:09 +00:00
|
|
|
#include "llvm/Support/ConvertUTF.h"
|
2019-07-09 18:10:36 +00:00
|
|
|
#include "llvm/Support/WindowsError.h"
|
2014-08-27 20:15:30 +00:00
|
|
|
|
2016-12-15 15:00:41 +00:00
|
|
|
#include <psapi.h>
|
2014-09-11 22:22:16 +00:00
|
|
|
|
2014-08-27 20:15:30 +00:00
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
2014-10-14 21:55:08 +00:00
|
|
|
namespace {
|
|
|
|
|
struct MonitorInfo {
|
2016-05-11 16:59:04 +00:00
|
|
|
Host::MonitorChildProcessCallback callback;
|
2014-10-14 21:55:08 +00:00
|
|
|
HANDLE process_handle;
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-30 16:56:40 +00:00
|
|
|
HostProcessWindows::HostProcessWindows()
|
|
|
|
|
: HostNativeProcessBase(), m_owns_handle(true) {}
|
|
|
|
|
|
2014-09-11 22:22:16 +00:00
|
|
|
HostProcessWindows::HostProcessWindows(lldb::process_t process)
|
|
|
|
|
: HostNativeProcessBase(process), m_owns_handle(true) {}
|
2016-09-06 20:57:50 +00:00
|
|
|
|
2014-08-27 20:15:30 +00:00
|
|
|
HostProcessWindows::~HostProcessWindows() { Close(); }
|
2016-09-06 20:57:50 +00:00
|
|
|
|
2014-11-05 22:16:28 +00:00
|
|
|
void HostProcessWindows::SetOwnsHandle(bool owns) { m_owns_handle = owns; }
|
2016-09-06 20:57:50 +00:00
|
|
|
|
2017-05-12 04:51:55 +00:00
|
|
|
Status HostProcessWindows::Terminate() {
|
|
|
|
|
Status error;
|
2014-09-11 22:22:16 +00:00
|
|
|
if (m_process == nullptr)
|
2014-08-27 20:15:30 +00:00
|
|
|
error.SetError(ERROR_INVALID_HANDLE, lldb::eErrorTypeWin32);
|
2016-09-06 20:57:50 +00:00
|
|
|
|
2014-08-27 20:15:30 +00:00
|
|
|
if (!::TerminateProcess(m_process, 0))
|
|
|
|
|
error.SetError(::GetLastError(), lldb::eErrorTypeWin32);
|
2016-09-06 20:57:50 +00:00
|
|
|
|
2014-08-27 20:15:30 +00:00
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-12 04:51:55 +00:00
|
|
|
Status HostProcessWindows::GetMainModule(FileSpec &file_spec) const {
|
|
|
|
|
Status error;
|
2014-08-27 20:15:30 +00:00
|
|
|
if (m_process == nullptr)
|
|
|
|
|
error.SetError(ERROR_INVALID_HANDLE, lldb::eErrorTypeWin32);
|
|
|
|
|
|
|
|
|
|
std::vector<wchar_t> wpath(PATH_MAX);
|
|
|
|
|
if (::GetProcessImageFileNameW(m_process, wpath.data(), wpath.size())) {
|
2016-03-22 17:58:09 +00:00
|
|
|
std::string path;
|
|
|
|
|
if (llvm::convertWideToUTF8(wpath.data(), path))
|
2018-11-02 08:47:33 +00:00
|
|
|
file_spec.SetFile(path, FileSpec::Style::native);
|
2014-08-27 20:15:30 +00:00
|
|
|
else
|
2016-03-22 17:58:09 +00:00
|
|
|
error.SetErrorString("Error converting path to UTF-8");
|
2016-09-06 20:57:50 +00:00
|
|
|
} else
|
2014-08-27 20:15:30 +00:00
|
|
|
error.SetError(::GetLastError(), lldb::eErrorTypeWin32);
|
|
|
|
|
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
lldb::pid_t HostProcessWindows::GetProcessId() const {
|
2014-10-14 21:55:08 +00:00
|
|
|
return (m_process == LLDB_INVALID_PROCESS) ? -1 : ::GetProcessId(m_process);
|
2014-08-27 20:15:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool HostProcessWindows::IsRunning() const {
|
2014-09-11 22:22:16 +00:00
|
|
|
if (m_process == nullptr)
|
2014-08-27 20:15:30 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
DWORD code = 0;
|
|
|
|
|
if (!::GetExitCodeProcess(m_process, &code))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return (code == STILL_ACTIVE);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-08 17:45:11 +00:00
|
|
|
llvm::Expected<HostThread> HostProcessWindows::StartMonitoring(
|
2016-05-11 16:59:04 +00:00
|
|
|
const Host::MonitorChildProcessCallback &callback, bool monitor_signals) {
|
2014-10-14 21:55:08 +00:00
|
|
|
MonitorInfo *info = new MonitorInfo;
|
|
|
|
|
info->callback = callback;
|
|
|
|
|
|
|
|
|
|
// Since the life of this HostProcessWindows instance and the life of the
|
2018-04-30 16:49:04 +00:00
|
|
|
// process may be different, duplicate the handle so that the monitor thread
|
|
|
|
|
// can have ownership over its own copy of the handle.
|
2014-10-14 21:55:08 +00:00
|
|
|
if (::DuplicateHandle(GetCurrentProcess(), m_process, GetCurrentProcess(),
|
2019-07-08 22:09:08 +00:00
|
|
|
&info->process_handle, 0, FALSE, DUPLICATE_SAME_ACCESS)) {
|
|
|
|
|
return ThreadLauncher::LaunchThread("ChildProcessMonitor",
|
|
|
|
|
HostProcessWindows::MonitorThread,
|
|
|
|
|
info);
|
|
|
|
|
} else {
|
2019-07-09 18:10:36 +00:00
|
|
|
return llvm::errorCodeToError(llvm::mapWindowsError(GetLastError()));
|
2019-07-08 22:09:08 +00:00
|
|
|
}
|
2014-10-14 21:55:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
lldb::thread_result_t HostProcessWindows::MonitorThread(void *thread_arg) {
|
|
|
|
|
DWORD exit_code;
|
|
|
|
|
|
|
|
|
|
MonitorInfo *info = static_cast<MonitorInfo *>(thread_arg);
|
|
|
|
|
if (info) {
|
2015-04-02 20:57:38 +00:00
|
|
|
::WaitForSingleObject(info->process_handle, INFINITE);
|
2014-10-14 21:55:08 +00:00
|
|
|
::GetExitCodeProcess(info->process_handle, &exit_code);
|
2016-05-11 16:59:04 +00:00
|
|
|
info->callback(::GetProcessId(info->process_handle), true, 0, exit_code);
|
2014-10-24 17:51:56 +00:00
|
|
|
::CloseHandle(info->process_handle);
|
2014-10-14 21:55:08 +00:00
|
|
|
delete (info);
|
|
|
|
|
}
|
2019-05-23 15:17:39 +00:00
|
|
|
return {};
|
2014-10-14 21:55:08 +00:00
|
|
|
}
|
|
|
|
|
|
2014-08-27 20:15:30 +00:00
|
|
|
void HostProcessWindows::Close() {
|
2014-11-05 22:16:28 +00:00
|
|
|
if (m_owns_handle && m_process != LLDB_INVALID_PROCESS)
|
2014-08-27 20:15:30 +00:00
|
|
|
::CloseHandle(m_process);
|
|
|
|
|
m_process = nullptr;
|
|
|
|
|
}
|