10#include "lldb/Host/Config.h"
14#include "llvm/Config/llvm-config.h"
15#include "llvm/Support/WindowsError.h"
31static DWORD
ToTimeout(std::optional<MainLoopWindows::TimePoint> point) {
32 using namespace std::chrono;
37 nanoseconds dur = (std::max)(*point - steady_clock::now(), nanoseconds(0));
38 return ceil<milliseconds>(dur).count();
45 explicit PipeEvent(
HANDLE handle)
46 : IOEvent(CreateEventW(nullptr, TRUE,
49 m_ready(CreateEventW(nullptr, TRUE,
51 assert(m_event && m_ready);
52 m_monitor_thread = std::thread(&PipeEvent::Monitor,
this);
55 ~PipeEvent()
override {
56 if (m_monitor_thread.joinable()) {
58 std::lock_guard<std::mutex> guard(m_mutex);
61 CancelIoEx(m_handle, &m_ov);
63 m_monitor_thread.join();
69 void WillPoll()
override {
70 std::lock_guard<std::mutex> guard(m_mutex);
72 HANDLE handles[2] = {m_event, m_ready};
73 if (WaitForMultipleObjects(2, handles, FALSE,
85 void Disarm()
override {
86 std::lock_guard<std::mutex> guard(m_mutex);
94 WaitForSingleObject(m_ready, INFINITE);
99 ZeroMemory(&m_ov,
sizeof(m_ov));
103 BOOL success = ReadFile(m_handle, buf, 0,
105 DWORD bytes_available = 0;
106 DWORD err = GetLastError();
107 if (!success && err == ERROR_IO_PENDING) {
108 success = GetOverlappedResult(m_handle, &m_ov, &bytes_read,
110 err = GetLastError();
113 success = PeekNamedPipe(m_handle,
nullptr, 0,
nullptr, &bytes_available,
115 err = GetLastError();
118 if (bytes_available == 0) {
122 }
else if (err == ERROR_NO_DATA) {
126 }
else if (err == ERROR_OPERATION_ABORTED) {
131 std::lock_guard<std::mutex> guard(m_mutex);
146 WaitForSingleObject(m_ready, INFINITE);
147 }
while (!m_stopped);
154 std::thread m_monitor_thread;
155 std::atomic<bool> m_stopped =
false;
161 explicit SocketEvent(SOCKET socket)
162 : IOEvent(WSACreateEvent()), m_socket(socket) {
163 assert(m_event != WSA_INVALID_EVENT);
166 ~SocketEvent()
override { WSACloseEvent(m_event); }
168 void WillPoll()
override {
170 WSAEventSelect(m_socket, m_event, FD_READ | FD_ACCEPT | FD_CLOSE);
175 void DidPoll()
override {
176 int result = WSAEventSelect(m_socket, WSA_INVALID_EVENT, 0);
181 void Disarm()
override { WSAResetEvent(m_event); }
196 assert(result == TRUE);
201 std::vector<HANDLE> events;
204 fd_info.event->WillPoll();
205 events.push_back(fd_info.event->GetHandle());
210 WSAWaitForMultipleEvents(events.size(), events.data(), FALSE,
214 fd_info.event->DidPoll();
216 if (result >= WSA_WAIT_EVENT_0 && result < WSA_WAIT_EVENT_0 + events.size())
217 return result - WSA_WAIT_EVENT_0;
220 if (result == WSA_WAIT_TIMEOUT)
221 return events.size() - 1;
223 return llvm::createStringError(llvm::inconvertibleErrorCode(),
224 "WSAWaitForMultipleEvents failed");
230 if (!object_sp || !object_sp->IsValid()) {
240 "File descriptor %p already monitored.", waitable_handle);
246 std::make_unique<SocketEvent>(
247 reinterpret_cast<SOCKET
>(waitable_handle)),
250 DWORD file_type = GetFileType(waitable_handle);
251 if (file_type != FILE_TYPE_CHAR && file_type != FILE_TYPE_PIPE) {
257 m_read_fds[waitable_handle] = {std::make_unique<PipeEvent>(waitable_handle),
276 llvm::Expected<size_t> signaled_event =
Poll();
281 auto &KV = *std::next(
m_read_fds.begin(), *signaled_event);
282 KV.second.event->Disarm();
283 KV.second.callback(*
this);
static llvm::raw_ostream & error(Stream &strm)
static DWORD ToTimeout(std::optional< MainLoopWindows::TimePoint > point)
static const WaitableHandle kInvalidHandleValue
lldb::file_t WaitableHandle
std::unique_ptr< ReadHandle > ReadHandleUP
std::optional< TimePoint > GetNextWakeupTime()
ReadHandleUP CreateReadHandle(const lldb::IOObjectSP &object_sp)
std::function< void(MainLoopBase &)> Callback
llvm::Expected< size_t > Poll()
~MainLoopWindows() override
llvm::DenseMap< IOObject::WaitableHandle, FdInfo > m_read_fds
bool Interrupt() override
Interrupt the loop that is currently waiting for events.
ReadHandleUP RegisterReadObject(const lldb::IOObjectSP &object_sp, const Callback &callback, Status &error) override
void UnregisterReadObject(IOObject::WaitableHandle handle) override
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
static Status FromErrorString(const char *str)
static Status FromError(llvm::Error error)
Avoid using this in new code. Migrate APIs to llvm::Expected instead.
#define UNUSED_IF_ASSERT_DISABLED(x)
A class that represents a running process on the host machine.
std::shared_ptr< lldb_private::IOObject > IOObjectSP