LLDB mainline
MainLoopWindows.cpp
Go to the documentation of this file.
1//===-- MainLoopWindows.cpp -----------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
10#include "lldb/Host/Config.h"
11#include "lldb/Host/Socket.h"
13#include "lldb/Utility/Status.h"
14#include "llvm/Config/llvm-config.h"
15#include "llvm/Support/WindowsError.h"
16#include <algorithm>
17#include <atomic>
18#include <cassert>
19#include <ctime>
20#include <io.h>
21#include <synchapi.h>
22#include <thread>
23#include <vector>
24#include <winbase.h>
25#include <winerror.h>
26#include <winsock2.h>
27
28using namespace lldb;
29using namespace lldb_private;
30
31static DWORD ToTimeout(std::optional<MainLoopWindows::TimePoint> point) {
32 using namespace std::chrono;
33
34 if (!point)
35 return WSA_INFINITE;
36
37 nanoseconds dur = (std::max)(*point - steady_clock::now(), nanoseconds(0));
38 return ceil<milliseconds>(dur).count();
39}
40
41namespace {
42
43class PipeEvent : public MainLoopWindows::IOEvent {
44public:
45 explicit PipeEvent(HANDLE handle)
46 : IOEvent(CreateEventW(nullptr, /*bManualReset=*/TRUE,
47 /*bInitialState=*/FALSE, nullptr)),
48 m_handle(handle),
49 m_ready(CreateEventW(nullptr, /*bManualReset=*/TRUE,
50 /*bInitialState=*/FALSE, nullptr)) {
51 assert(m_event && m_ready);
52 m_monitor_thread = std::thread(&PipeEvent::Monitor, this);
53 }
54
55 ~PipeEvent() override {
56 if (m_monitor_thread.joinable()) {
57 {
58 std::lock_guard<std::mutex> guard(m_mutex);
59 m_stopped = true;
60 SetEvent(m_ready);
61 CancelIoEx(m_handle, &m_ov);
62 }
63 m_monitor_thread.join();
64 }
65 CloseHandle(m_event);
66 CloseHandle(m_ready);
67 }
68
69 void WillPoll() override {
70 std::lock_guard<std::mutex> guard(m_mutex);
71
72 HANDLE handles[2] = {m_event, m_ready};
73 if (WaitForMultipleObjects(2, handles, /*bWaitAll=*/FALSE,
74 /*dwMilliseconds=*/0) != WAIT_TIMEOUT) {
75 // Either:
76 // - The thread has already signalled that the data is available. No need
77 // for further polling until we consume that event.
78 // - The thread is already waiting for data to become available.
79 return;
80 }
81 // Start waiting.
82 SetEvent(m_ready);
83 }
84
85 void Disarm() override {
86 std::lock_guard<std::mutex> guard(m_mutex);
87 ResetEvent(m_event);
88 }
89
90 /// Monitors the handle performing a zero byte read to determine when data is
91 /// avaiable.
92 void Monitor() {
93 // Wait until the MainLoop tells us to start.
94 WaitForSingleObject(m_ready, INFINITE);
95
96 do {
97 char buf[1];
98 DWORD bytes_read = 0;
99 ZeroMemory(&m_ov, sizeof(m_ov));
100 // Block on a 0-byte read; this will only resume when data is
101 // available in the pipe. The pipe must be PIPE_WAIT or this thread
102 // will spin.
103 BOOL success = ReadFile(m_handle, buf, /*nNumberOfBytesToRead=*/0,
104 &bytes_read, &m_ov);
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,
109 /*bWait=*/TRUE);
110 err = GetLastError();
111 }
112 if (success) {
113 success = PeekNamedPipe(m_handle, nullptr, 0, nullptr, &bytes_available,
114 nullptr);
115 err = GetLastError();
116 }
117 if (success) {
118 if (bytes_available == 0) {
119 // This can happen with a zero-byte write. Try again.
120 continue;
121 }
122 } else if (err == ERROR_NO_DATA) {
123 // The pipe is nonblocking. Try again.
124 Sleep(0);
125 continue;
126 } else if (err == ERROR_OPERATION_ABORTED) {
127 // Read may have been cancelled, try again.
128 continue;
129 }
130 {
131 std::lock_guard<std::mutex> guard(m_mutex);
132
133 // Notify that data is available on the pipe.
134 SetEvent(m_event);
135 if (m_stopped) {
136 // The destructor might have called SetEvent(m_ready) before this
137 // block. If that's the case, ResetEvent(m_ready) will cause
138 // WaitForSingleObject to wait forever unless we break early.
139 break;
140 }
141 // Stop polling until we're told to resume.
142 ResetEvent(m_ready);
143 }
144
145 // Wait until the current read is consumed before doing the next read.
146 WaitForSingleObject(m_ready, INFINITE);
147 } while (!m_stopped);
148 }
149
150private:
151 HANDLE m_handle;
152 HANDLE m_ready;
153 OVERLAPPED m_ov;
154 std::thread m_monitor_thread;
155 std::atomic<bool> m_stopped = false;
156 std::mutex m_mutex;
157};
158
159class SocketEvent : public MainLoopWindows::IOEvent {
160public:
161 explicit SocketEvent(SOCKET socket)
162 : IOEvent(WSACreateEvent()), m_socket(socket) {
163 assert(m_event != WSA_INVALID_EVENT);
164 }
165
166 ~SocketEvent() override { WSACloseEvent(m_event); }
167
168 void WillPoll() override {
169 int result =
170 WSAEventSelect(m_socket, m_event, FD_READ | FD_ACCEPT | FD_CLOSE);
171 assert(result == 0);
173 }
174
175 void DidPoll() override {
176 int result = WSAEventSelect(m_socket, WSA_INVALID_EVENT, 0);
177 assert(result == 0);
179 }
180
181 void Disarm() override { WSAResetEvent(m_event); }
182
183 SOCKET m_socket;
184};
185
186} // namespace
187
189 m_interrupt_event = WSACreateEvent();
190 assert(m_interrupt_event != WSA_INVALID_EVENT);
191}
192
194 assert(m_read_fds.empty());
195 BOOL result = WSACloseEvent(m_interrupt_event);
196 assert(result == TRUE);
198}
199
200llvm::Expected<size_t> MainLoopWindows::Poll() {
201 std::vector<HANDLE> events;
202 events.reserve(m_read_fds.size() + 1);
203 for (auto &[_, fd_info] : m_read_fds) {
204 fd_info.event->WillPoll();
205 events.push_back(fd_info.event->GetHandle());
206 }
207 events.push_back(m_interrupt_event);
208
209 DWORD result =
210 WSAWaitForMultipleEvents(events.size(), events.data(), FALSE,
211 ToTimeout(GetNextWakeupTime()), FALSE);
212
213 for (auto &[_, fd_info] : m_read_fds)
214 fd_info.event->DidPoll();
215
216 if (result >= WSA_WAIT_EVENT_0 && result < WSA_WAIT_EVENT_0 + events.size())
217 return result - WSA_WAIT_EVENT_0;
218
219 // A timeout is treated as a (premature) signalization of the interrupt event.
220 if (result == WSA_WAIT_TIMEOUT)
221 return events.size() - 1;
222
223 return llvm::createStringError(llvm::inconvertibleErrorCode(),
224 "WSAWaitForMultipleEvents failed");
225}
226
229 const Callback &callback, Status &error) {
230 if (!object_sp || !object_sp->IsValid()) {
231 error = Status::FromErrorString("IO object is not valid.");
232 return nullptr;
233 }
234
235 IOObject::WaitableHandle waitable_handle = object_sp->GetWaitableHandle();
236 assert(waitable_handle != IOObject::kInvalidHandleValue);
237
238 if (m_read_fds.find(waitable_handle) != m_read_fds.end()) {
240 "File descriptor %p already monitored.", waitable_handle);
241 return nullptr;
242 }
243
244 if (object_sp->GetFdType() == IOObject::eFDTypeSocket) {
245 m_read_fds[waitable_handle] = {
246 std::make_unique<SocketEvent>(
247 reinterpret_cast<SOCKET>(waitable_handle)),
248 callback};
249 } else {
250 DWORD file_type = GetFileType(waitable_handle);
251 if (file_type != FILE_TYPE_CHAR && file_type != FILE_TYPE_PIPE) {
252 error = Status::FromErrorStringWithFormat("Unsupported file type %ld",
253 file_type);
254 return nullptr;
255 }
256
257 m_read_fds[waitable_handle] = {std::make_unique<PipeEvent>(waitable_handle),
258 callback};
259 }
260
261 return CreateReadHandle(object_sp);
262}
263
265 auto it = m_read_fds.find(handle);
266 assert(it != m_read_fds.end());
267 m_read_fds.erase(it);
268}
269
271 m_terminate_request = false;
272
274
275 while (!m_terminate_request) {
276 llvm::Expected<size_t> signaled_event = Poll();
277 if (!signaled_event)
278 return Status::FromError(signaled_event.takeError());
279
280 if (*signaled_event < m_read_fds.size()) {
281 auto &KV = *std::next(m_read_fds.begin(), *signaled_event);
282 KV.second.event->Disarm();
283 KV.second.callback(*this); // Do the work.
284 } else {
285 assert(*signaled_event == m_read_fds.size());
286 WSAResetEvent(m_interrupt_event);
287 }
289 }
290 return Status();
291}
292
294 return WSASetEvent(m_interrupt_event);
295}
static llvm::raw_ostream & error(Stream &strm)
static DWORD ToTimeout(std::optional< MainLoopWindows::TimePoint > point)
void * HANDLE
static const WaitableHandle kInvalidHandleValue
Definition IOObject.h:31
lldb::file_t WaitableHandle
Definition IOObject.h:29
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()
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
An error handling class.
Definition Status.h:118
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
Definition Status.cpp:106
static Status FromErrorString(const char *str)
Definition Status.h:141
static Status FromError(llvm::Error error)
Avoid using this in new code. Migrate APIs to llvm::Expected instead.
Definition Status.cpp:136
#define UNUSED_IF_ASSERT_DISABLED(x)
A class that represents a running process on the host machine.
std::shared_ptr< lldb_private::IOObject > IOObjectSP