-
Notifications
You must be signed in to change notification settings - Fork 465
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Switch Win32 pipes to PIPE_WAIT with sentinel bufsize #854
base: main
Are you sure you want to change the base?
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm comfortable with this fix
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for looking at this! Just a comment, no opinion on the change.
@@ -277,6 +277,9 @@ _dispatch_pipe_monitor_thread(void *context) | |||
char cBuffer[1]; | |||
DWORD dwNumberOfBytesTransferred; | |||
OVERLAPPED ov = {0}; | |||
// Block on a 0-byte read; this will only resume when data is | |||
// available in the pipe. The pipe must be PIPE_WAIT or this thread | |||
// will spin. | |||
BOOL bSuccess = ReadFile(hPipe, cBuffer, /* nNumberOfBytesToRead */ 0, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why not do a WaitForSingleObjectEx()? In particular pay attention to the bAlertable condition.
IMHO, I don't see the need to use a thread, WaitForMultipleObjectsEx() would do just fine and be more performant.
Note that I don't have a good understanding of this code base so my humble opinion may be wrong but I have some experience with win32 pipes.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure I follow. I looked through the docs of WaitForSingleObjectEx and I don't see where it applies here. You can't wait on a pipe, no?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Effectively. The idea is to wait on the events that are connected to the pending overlappedio.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Expanding; https://learn.microsoft.com/en-us/windows/win32/devio/overlapped-operations
The OVERLAPPED structure must contain a handle to a manual-reset (not an auto-reset) event object
https://learn.microsoft.com/en-us/windows/win32/api/minwinbase/ns-minwinbase-overlapped
especially the section about hEvent.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Or said differently, I didn't recall passing an OVERLAPPED structure with a null hEvent to be valid.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In general we cannot rely on Overlapped I/O here (which would simplify many things). Pipes are created externally and their handles are passed into libdispatch, and there's no way to make a pipe FILE_FLAG_OVERLAPPED
after the fact (at least, none that I could find, and I looked for this).
We also need to support anonymous pipes, which do not support Overlapped I/O. This requirement is embedded in the libdispatch tests, which use anonymous pipes.
It could be I'm misunderstanding something about your proposal though. After discounting Overlapped I/O because the requirements mentioned above, I stopped looking into that solution space, so am less familiar with the APIs you are referencing.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh ok that's a bummer. Sorry I didn't mean to send you off track. From https://learn.microsoft.com/en-us/windows/win32/ipc/anonymous-pipe-operations:
In addition, the lpOverlapped parameter of ReadFile and WriteFile is ignored when these functions are used with anonymous pipes.
Emphasis mine. That's why the code works. I'd recommend to pass NULL instead or &ov to avoid confusion. Other than that, lgtm.
b0ea38b
to
93c4bcd
Compare
93c4bcd
to
b2544fe
Compare
Fixes #820 . There is a lot of useful context in this issue, which I will partially reproduce below.
I have an alternate PR that fixes this problem in a less fundamental way, but without a small requirements change to libdispatch which I describe below: #853.
Problem description, copied from the aforementioned PR:
The real issue here is the inability to distinguish on the write side of a pipe between the following cases:
This PR implements the same switch back to
PIPE_WAIT
as in #853, but uses a trick to enableWriteQuotaAvailable == 1
to be used as a sentinel value to distinguish the "output buffer is full" case. By always leaving a free byte of space in the buffer when writing, the only way thatWriteQuotaAvailable == 0
can happen is if a reader has requested a full buffer's worth of data.This means that when we see
WriteQuotaAvailable == 1
, we know the output buffer is "full" and not attempt to perform a blocking write.The downside to this approach is that we can no longer serially write-then-read a full buffer's worth of data, because the write will not finish writing the last byte until reading has commenced. I don't see this scenario ever arising in practice: if you are trying to move data serially from one buffer to another on the same thread, there are better ways to do that than through a libdispatch pipe. But regardless, this requirement is implicit due to the serial nature of the
dispatch_io_pipe
tests. If if is acceptable to relax this requirement from "writes of<=bufSize
are guaranteed to complete in the absence of a reader" to "writes of<bufSize
are guaranteed to complete in the absence of a reader" then I think this solution is more robust than just #853.It's entirely possible I'm missing some good reason why this requirement is there though, in which case I think #853 should be acceptable.
cc @compnerd