- fixed ConnectNamedPipe to properly wait on completion for synchronous operations

- prevent completion port notification by not passing an APC context to the kernel when the low-order bit of the event handle is set (for asynchronous operations)

svn path=/trunk/; revision=15048
This commit is contained in:
Thomas Bluemel 2005-05-06 16:07:51 +00:00
parent f16e5b9e1d
commit ecba32de75

View file

@ -300,37 +300,53 @@ WaitNamedPipeW(LPCWSTR lpNamedPipeName,
* @implemented * @implemented
*/ */
BOOL STDCALL BOOL STDCALL
ConnectNamedPipe(HANDLE hNamedPipe, ConnectNamedPipe(IN HANDLE hNamedPipe,
LPOVERLAPPED lpOverlapped) IN LPOVERLAPPED lpOverlapped)
{ {
PIO_STATUS_BLOCK IoStatusBlock;
IO_STATUS_BLOCK Iosb;
HANDLE hEvent;
NTSTATUS Status; NTSTATUS Status;
if (lpOverlapped != NULL) if (lpOverlapped != NULL)
{ {
PVOID ApcContext;
lpOverlapped->Internal = STATUS_PENDING; lpOverlapped->Internal = STATUS_PENDING;
hEvent = lpOverlapped->hEvent; ApcContext = (((ULONG_PTR)lpOverlapped->hEvent & 0x1) ? NULL : lpOverlapped);
IoStatusBlock = (PIO_STATUS_BLOCK)lpOverlapped;
}
else
{
IoStatusBlock = &Iosb;
hEvent = NULL;
}
Status = NtFsControlFile(hNamedPipe, Status = NtFsControlFile(hNamedPipe,
hEvent, lpOverlapped->hEvent,
NULL, NULL,
NULL, ApcContext,
IoStatusBlock, (PIO_STATUS_BLOCK)lpOverlapped,
FSCTL_PIPE_LISTEN, FSCTL_PIPE_LISTEN,
NULL, NULL,
0, 0,
NULL, NULL,
0); 0);
if ((lpOverlapped == NULL) && (Status == STATUS_PENDING))
/* return FALSE in case of failure and pending operations! */
if (!NT_SUCCESS(Status) || Status == STATUS_PENDING)
{
SetLastErrorByStatus(Status);
return FALSE;
}
}
else
{
IO_STATUS_BLOCK Iosb;
Status = NtFsControlFile(hNamedPipe,
NULL,
NULL,
NULL,
&Iosb,
FSCTL_PIPE_LISTEN,
NULL,
0,
NULL,
0);
/* wait in case operation is pending */
if (Status == STATUS_PENDING)
{ {
Status = NtWaitForSingleObject(hNamedPipe, Status = NtWaitForSingleObject(hNamedPipe,
FALSE, FALSE,
@ -341,12 +357,12 @@ ConnectNamedPipe(HANDLE hNamedPipe,
} }
} }
if ((!NT_SUCCESS(Status) && Status != STATUS_PIPE_CONNECTED) || if (!NT_SUCCESS(Status))
(Status == STATUS_PENDING))
{ {
SetLastErrorByStatus(Status); SetLastErrorByStatus(Status);
return FALSE; return FALSE;
} }
}
return TRUE; return TRUE;
} }