ntdll: Avoid setting status in IO_STATUS_BLOCK until the I/O operation is done.

This commit is contained in:
Alexandre Julliard 2007-04-05 10:55:40 +02:00
parent 0ce39d5869
commit e57749d5a8
1 changed files with 125 additions and 133 deletions

View File

@ -250,13 +250,13 @@ typedef struct async_fileio
HANDLE event; HANDLE event;
} async_fileio; } async_fileio;
static void fileio_terminate(async_fileio *fileio, IO_STATUS_BLOCK* iosb) static void fileio_terminate(async_fileio *fileio, IO_STATUS_BLOCK* iosb, NTSTATUS status)
{ {
TRACE("data: %p\n", fileio); TRACE("data: %p\n", fileio);
if (fileio->apc && iosb->u.Status = status;
(iosb->u.Status == STATUS_SUCCESS || fileio->queue_apc_on_error)) if (fileio->apc && (status == STATUS_SUCCESS || fileio->queue_apc_on_error))
fileio->apc( fileio->apc_user, iosb, iosb->Information ); fileio->apc( fileio->apc_user, iosb, 0 );
RtlFreeHeap( GetProcessHeap(), 0, fileio ); RtlFreeHeap( GetProcessHeap(), 0, fileio );
} }
@ -283,11 +283,9 @@ static ULONG fileio_queue_async(async_fileio* fileio, IO_STATUS_BLOCK* iosb,
SERVER_END_REQ; SERVER_END_REQ;
if (status != STATUS_PENDING) if (status != STATUS_PENDING)
{ fileio_terminate( fileio, iosb, status );
iosb->u.Status = status; else
(apc)( fileio, iosb, iosb->u.Status ); NtCurrentTeb()->num_async_io++;
}
else NtCurrentTeb()->num_async_io++;
return status; return status;
} }
@ -351,12 +349,11 @@ static void WINAPI FILE_AsyncReadService(void *user, PIO_STATUS_BLOCK iosb, ULON
switch (status) switch (status)
{ {
case STATUS_ALERTED: /* got some new data */ case STATUS_ALERTED: /* got some new data */
if (iosb->u.Status != STATUS_PENDING) FIXME("unexpected status %08x\n", iosb->u.Status);
/* check to see if the data is ready (non-blocking) */ /* check to see if the data is ready (non-blocking) */
if ((iosb->u.Status = server_get_unix_fd( fileio->handle, FILE_READ_DATA, &fd, if ((status = server_get_unix_fd( fileio->handle, FILE_READ_DATA, &fd,
&needs_close, NULL, NULL ))) &needs_close, NULL, NULL )))
{ {
fileio_terminate(fileio, iosb); fileio_terminate(fileio, iosb, status);
break; break;
} }
result = read(fd, &fileio->buffer[already], fileio->count - already); result = read(fd, &fileio->buffer[already], fileio->count - already);
@ -367,42 +364,41 @@ static void WINAPI FILE_AsyncReadService(void *user, PIO_STATUS_BLOCK iosb, ULON
if (errno == EAGAIN || errno == EINTR) if (errno == EAGAIN || errno == EINTR)
{ {
TRACE("Deferred read %d\n", errno); TRACE("Deferred read %d\n", errno);
iosb->u.Status = STATUS_PENDING; status = STATUS_PENDING;
} }
else /* check to see if the transfer is complete */ else /* check to see if the transfer is complete */
iosb->u.Status = FILE_GetNtStatus(); status = FILE_GetNtStatus();
} }
else if (result == 0) else if (result == 0)
{ {
iosb->u.Status = iosb->Information ? STATUS_SUCCESS : STATUS_END_OF_FILE; status = iosb->Information ? STATUS_SUCCESS : STATUS_END_OF_FILE;
} }
else else
{ {
iosb->Information += result; iosb->Information += result;
if (iosb->Information >= fileio->count || fileio->avail_mode) if (iosb->Information >= fileio->count || fileio->avail_mode)
iosb->u.Status = STATUS_SUCCESS; status = STATUS_SUCCESS;
else else
{ {
/* if we only have to read the available data, and none is available, /* if we only have to read the available data, and none is available,
* simply cancel the request. If data was available, it has been read * simply cancel the request. If data was available, it has been read
* while in by previous call (NtDelayExecution) * while in by previous call (NtDelayExecution)
*/ */
iosb->u.Status = (fileio->avail_mode) ? STATUS_SUCCESS : STATUS_PENDING; status = (fileio->avail_mode) ? STATUS_SUCCESS : STATUS_PENDING;
} }
TRACE("read %d more bytes %ld/%d so far (%s)\n", TRACE("read %d more bytes %ld/%d so far (%s)\n",
result, iosb->Information, fileio->count, result, iosb->Information, fileio->count,
(iosb->u.Status == STATUS_SUCCESS) ? "success" : "pending"); (status == STATUS_SUCCESS) ? "success" : "pending");
} }
/* queue another async operation ? */ /* queue another async operation ? */
if (iosb->u.Status == STATUS_PENDING) if (status == STATUS_PENDING)
fileio_queue_async(fileio, iosb, TRUE); fileio_queue_async(fileio, iosb, TRUE);
else else
fileio_terminate(fileio, iosb); fileio_terminate(fileio, iosb, status);
break; break;
default: default:
iosb->u.Status = status; fileio_terminate(fileio, iosb, status);
fileio_terminate(fileio, iosb);
break; break;
} }
} }
@ -436,6 +432,7 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
PLARGE_INTEGER offset, PULONG key) PLARGE_INTEGER offset, PULONG key)
{ {
int result, unix_handle, needs_close, flags; int result, unix_handle, needs_close, flags;
NTSTATUS status;
enum server_fd_type type; enum server_fd_type type;
TRACE("(%p,%p,%p,%p,%p,%p,0x%08x,%p,%p),partial stub!\n", TRACE("(%p,%p,%p,%p,%p,%p,0x%08x,%p,%p),partial stub!\n",
@ -444,9 +441,13 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
if (!io_status) return STATUS_ACCESS_VIOLATION; if (!io_status) return STATUS_ACCESS_VIOLATION;
io_status->Information = 0; io_status->Information = 0;
io_status->u.Status = server_get_unix_fd( hFile, FILE_READ_DATA, &unix_handle, status = server_get_unix_fd( hFile, FILE_READ_DATA, &unix_handle,
&needs_close, &type, &flags ); &needs_close, &type, &flags );
if (io_status->u.Status) return io_status->u.Status; if (status)
{
io_status->u.Status = status;
return status;
}
if (type == FD_TYPE_FILE && offset) if (type == FD_TYPE_FILE && offset)
{ {
@ -458,19 +459,18 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
if (errno == EINTR) continue; if (errno == EINTR) continue;
if (errno == EAGAIN || errno == ESPIPE) if (errno == EAGAIN || errno == ESPIPE)
{ {
io_status->u.Status = STATUS_PENDING; status = STATUS_PENDING;
break; break;
} }
result = 0; result = 0;
io_status->u.Status = FILE_GetNtStatus(); status = FILE_GetNtStatus();
goto done; goto done;
} }
if (result >= 0) if (result >= 0)
{ {
io_status->Information = result; io_status->Information = result;
io_status->u.Status = result ? STATUS_SUCCESS : STATUS_END_OF_FILE; status = result ? STATUS_SUCCESS : STATUS_END_OF_FILE;
if (hEvent) NtSetEvent( hEvent, NULL ); if (hEvent) NtSetEvent( hEvent, NULL );
if (apc && !io_status->u.Status) apc( apc_user, io_status, io_status->Information );
goto done; goto done;
} }
} }
@ -478,7 +478,7 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
{ {
if (lseek( unix_handle, offset->QuadPart, SEEK_SET ) == (off_t)-1) if (lseek( unix_handle, offset->QuadPart, SEEK_SET ) == (off_t)-1)
{ {
io_status->u.Status = FILE_GetNtStatus(); status = FILE_GetNtStatus();
goto done; goto done;
} }
} }
@ -486,8 +486,8 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
if (flags & FD_FLAG_RECV_SHUTDOWN) if (flags & FD_FLAG_RECV_SHUTDOWN)
{ {
if (needs_close) close( unix_handle ); status = STATUS_PIPE_DISCONNECTED;
return STATUS_PIPE_DISCONNECTED; goto done;
} }
if (flags & FD_FLAG_TIMEOUT) if (flags & FD_FLAG_TIMEOUT)
@ -496,27 +496,22 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
{ {
/* this shouldn't happen, but check it */ /* this shouldn't happen, but check it */
FIXME("NIY-hEvent\n"); FIXME("NIY-hEvent\n");
if (needs_close) close( unix_handle ); status = STATUS_NOT_IMPLEMENTED;
return STATUS_NOT_IMPLEMENTED; goto done;
}
io_status->u.Status = NtCreateEvent(&hEvent, EVENT_ALL_ACCESS, NULL, 0, 0);
if (io_status->u.Status)
{
if (needs_close) close( unix_handle );
return io_status->u.Status;
} }
status = NtCreateEvent(&hEvent, EVENT_ALL_ACCESS, NULL, 0, 0);
if (status) goto done;
} }
if (flags & (FD_FLAG_OVERLAPPED|FD_FLAG_TIMEOUT)) if (flags & (FD_FLAG_OVERLAPPED|FD_FLAG_TIMEOUT))
{ {
async_fileio* fileio; async_fileio* fileio;
NTSTATUS ret;
if (!(fileio = RtlAllocateHeap(GetProcessHeap(), 0, sizeof(async_fileio)))) if (!(fileio = RtlAllocateHeap(GetProcessHeap(), 0, sizeof(async_fileio))))
{ {
if (needs_close) close( unix_handle );
if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent); if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent);
return STATUS_NO_MEMORY; status = STATUS_NO_MEMORY;
goto done;
} }
fileio->handle = hFile; fileio->handle = hFile;
fileio->count = length; fileio->count = length;
@ -526,25 +521,19 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
fileio->queue_apc_on_error = 0; fileio->queue_apc_on_error = 0;
fileio->avail_mode = (flags & FD_FLAG_AVAILABLE); fileio->avail_mode = (flags & FD_FLAG_AVAILABLE);
fileio->event = hEvent; fileio->event = hEvent;
if (needs_close) close( unix_handle );
io_status->u.Status = STATUS_PENDING; io_status->u.Status = STATUS_PENDING;
ret = fileio_queue_async(fileio, io_status, TRUE); status = fileio_queue_async(fileio, io_status, TRUE);
if (ret != STATUS_PENDING) if (status != STATUS_PENDING)
{ {
if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent); if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent);
return ret; goto done;
} }
if (needs_close) close( unix_handle );
if (flags & FD_FLAG_TIMEOUT) if (flags & FD_FLAG_TIMEOUT)
{ {
do while (NtWaitForSingleObject(hEvent, TRUE, NULL) == STATUS_USER_APC) /* nothing */ ;
{
ret = NtWaitForSingleObject(hEvent, TRUE, NULL);
}
while (ret == STATUS_USER_APC && io_status->u.Status == STATUS_PENDING);
NtClose(hEvent); NtClose(hEvent);
if (ret != STATUS_USER_APC)
fileio->queue_apc_on_error = 1;
} }
else else
{ {
@ -552,13 +541,13 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
/* let some APC be run, this will read some already pending data */ /* let some APC be run, this will read some already pending data */
timeout.u.LowPart = timeout.u.HighPart = 0; timeout.u.LowPart = timeout.u.HighPart = 0;
ret = NtDelayExecution( TRUE, &timeout ); status = NtDelayExecution( TRUE, &timeout );
/* the apc didn't run and therefore the completion routine now /* the apc didn't run and therefore the completion routine now
* needs to be sent errors. * needs to be sent errors.
* Note that there is no race between setting this flag and * Note that there is no race between setting this flag and
* returning errors because apc's are run only during alertable * returning errors because apc's are run only during alertable
* waits */ * waits */
if (ret != STATUS_USER_APC) if (status != STATUS_USER_APC)
fileio->queue_apc_on_error = 1; fileio->queue_apc_on_error = 1;
} }
TRACE("= 0x%08x\n", io_status->u.Status); TRACE("= 0x%08x\n", io_status->u.Status);
@ -566,26 +555,29 @@ NTSTATUS WINAPI NtReadFile(HANDLE hFile, HANDLE hEvent,
} }
/* code for synchronous reads */ /* code for synchronous reads */
status = STATUS_SUCCESS;
while ((result = read( unix_handle, buffer, length )) == -1) while ((result = read( unix_handle, buffer, length )) == -1)
{ {
if ((errno == EAGAIN) || (errno == EINTR)) continue; if ((errno == EAGAIN) || (errno == EINTR)) continue;
io_status->u.Status = FILE_GetNtStatus(); status = FILE_GetNtStatus();
result = 0; result = 0;
break; break;
} }
io_status->Information = result; io_status->Information = result;
if (io_status->u.Status == STATUS_SUCCESS && io_status->Information == 0) if (status == STATUS_SUCCESS && io_status->Information == 0)
{ {
struct stat st; struct stat st;
if (fstat( unix_handle, &st ) != -1 && S_ISSOCK( st.st_mode )) if (fstat( unix_handle, &st ) != -1 && S_ISSOCK( st.st_mode ))
io_status->u.Status = STATUS_PIPE_BROKEN; status = STATUS_PIPE_BROKEN;
else else
io_status->u.Status = STATUS_END_OF_FILE; status = STATUS_END_OF_FILE;
} }
done: done:
if (needs_close) close( unix_handle ); if (needs_close) close( unix_handle );
TRACE("= 0x%08x (%lu)\n", io_status->u.Status, io_status->Information); TRACE("= 0x%08x (%lu)\n", status, io_status->Information);
return io_status->u.Status; if (status != STATUS_PENDING) io_status->u.Status = status;
if (apc && !status) apc( apc_user, io_status, 0 );
return status;
} }
/*********************************************************************** /***********************************************************************
@ -603,10 +595,10 @@ static void WINAPI FILE_AsyncWriteService(void *ovp, IO_STATUS_BLOCK *iosb, ULON
{ {
case STATUS_ALERTED: case STATUS_ALERTED:
/* write some data (non-blocking) */ /* write some data (non-blocking) */
if ((iosb->u.Status = server_get_unix_fd( fileio->handle, FILE_WRITE_DATA, &fd, if ((status = server_get_unix_fd( fileio->handle, FILE_WRITE_DATA, &fd,
&needs_close, NULL, NULL ))) &needs_close, NULL, NULL )))
{ {
fileio_terminate(fileio, iosb); fileio_terminate(fileio, iosb, status);
break; break;
} }
result = write(fd, &fileio->buffer[already], fileio->count - already); result = write(fd, &fileio->buffer[already], fileio->count - already);
@ -614,24 +606,23 @@ static void WINAPI FILE_AsyncWriteService(void *ovp, IO_STATUS_BLOCK *iosb, ULON
if (result < 0) if (result < 0)
{ {
if (errno == EAGAIN || errno == EINTR) iosb->u.Status = STATUS_PENDING; if (errno == EAGAIN || errno == EINTR) status = STATUS_PENDING;
else iosb->u.Status = FILE_GetNtStatus(); else status = FILE_GetNtStatus();
} }
else else
{ {
iosb->Information += result; iosb->Information += result;
iosb->u.Status = (iosb->Information < fileio->count) ? STATUS_PENDING : STATUS_SUCCESS; status = (iosb->Information < fileio->count) ? STATUS_PENDING : STATUS_SUCCESS;
TRACE("wrote %d more bytes %ld/%d so far\n", TRACE("wrote %d more bytes %ld/%d so far\n",
result, iosb->Information, fileio->count); result, iosb->Information, fileio->count);
} }
if (iosb->u.Status == STATUS_PENDING) if (status == STATUS_PENDING)
fileio_queue_async(fileio, iosb, FALSE); fileio_queue_async(fileio, iosb, FALSE);
else else
fileio_terminate(fileio, iosb); fileio_terminate(fileio, iosb, status);
break; break;
default: default:
iosb->u.Status = status; fileio_terminate(fileio, iosb, status);
fileio_terminate(fileio, iosb);
break; break;
} }
} }
@ -665,6 +656,7 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
PLARGE_INTEGER offset, PULONG key) PLARGE_INTEGER offset, PULONG key)
{ {
int result, unix_handle, needs_close, flags; int result, unix_handle, needs_close, flags;
NTSTATUS status;
enum server_fd_type type; enum server_fd_type type;
TRACE("(%p,%p,%p,%p,%p,%p,0x%08x,%p,%p)!\n", TRACE("(%p,%p,%p,%p,%p,%p,0x%08x,%p,%p)!\n",
@ -673,9 +665,13 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
if (!io_status) return STATUS_ACCESS_VIOLATION; if (!io_status) return STATUS_ACCESS_VIOLATION;
io_status->Information = 0; io_status->Information = 0;
io_status->u.Status = server_get_unix_fd( hFile, FILE_WRITE_DATA, &unix_handle, status = server_get_unix_fd( hFile, FILE_WRITE_DATA, &unix_handle,
&needs_close, &type, &flags ); &needs_close, &type, &flags );
if (io_status->u.Status) return io_status->u.Status; if (status)
{
io_status->u.Status = status;
return status;
}
if (type == FD_TYPE_FILE && offset) if (type == FD_TYPE_FILE && offset)
{ {
@ -687,20 +683,19 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
if (errno == EINTR) continue; if (errno == EINTR) continue;
if (errno == EAGAIN || errno == ESPIPE) if (errno == EAGAIN || errno == ESPIPE)
{ {
io_status->u.Status = STATUS_PENDING; status = STATUS_PENDING;
break; break;
} }
result = 0; result = 0;
if (errno == EFAULT) io_status->u.Status = STATUS_INVALID_USER_BUFFER; if (errno == EFAULT) status = STATUS_INVALID_USER_BUFFER;
else io_status->u.Status = FILE_GetNtStatus(); else status = FILE_GetNtStatus();
goto done; goto done;
} }
if (result >= 0) if (result >= 0)
{ {
io_status->Information = result; io_status->Information = result;
io_status->u.Status = STATUS_SUCCESS; status = STATUS_SUCCESS;
if (hEvent) NtSetEvent( hEvent, NULL ); if (hEvent) NtSetEvent( hEvent, NULL );
if (apc) apc( apc_user, io_status, io_status->Information );
goto done; goto done;
} }
} }
@ -708,7 +703,7 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
{ {
if (lseek( unix_handle, offset->QuadPart, SEEK_SET ) == (off_t)-1) if (lseek( unix_handle, offset->QuadPart, SEEK_SET ) == (off_t)-1)
{ {
io_status->u.Status = FILE_GetNtStatus(); status = FILE_GetNtStatus();
goto done; goto done;
} }
} }
@ -716,8 +711,8 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
if (flags & FD_FLAG_SEND_SHUTDOWN) if (flags & FD_FLAG_SEND_SHUTDOWN)
{ {
if (needs_close) close( unix_handle ); status = STATUS_PIPE_DISCONNECTED;
return STATUS_PIPE_DISCONNECTED; goto done;
} }
if (flags & FD_FLAG_TIMEOUT) if (flags & FD_FLAG_TIMEOUT)
@ -726,27 +721,22 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
{ {
/* this shouldn't happen, but check it */ /* this shouldn't happen, but check it */
FIXME("NIY-hEvent\n"); FIXME("NIY-hEvent\n");
if (needs_close) close( unix_handle ); status = STATUS_NOT_IMPLEMENTED;
return STATUS_NOT_IMPLEMENTED; goto done;
}
io_status->u.Status = NtCreateEvent(&hEvent, EVENT_ALL_ACCESS, NULL, 0, 0);
if (io_status->u.Status)
{
if (needs_close) close( unix_handle );
return io_status->u.Status;
} }
status = NtCreateEvent(&hEvent, EVENT_ALL_ACCESS, NULL, 0, 0);
if (status) goto done;
} }
if (flags & (FD_FLAG_OVERLAPPED|FD_FLAG_TIMEOUT)) if (flags & (FD_FLAG_OVERLAPPED|FD_FLAG_TIMEOUT))
{ {
async_fileio* fileio; async_fileio* fileio;
NTSTATUS ret;
if (!(fileio = RtlAllocateHeap(GetProcessHeap(), 0, sizeof(async_fileio)))) if (!(fileio = RtlAllocateHeap(GetProcessHeap(), 0, sizeof(async_fileio))))
{ {
if (needs_close) close( unix_handle );
if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent); if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent);
return STATUS_NO_MEMORY; status = STATUS_NO_MEMORY;
goto done;
} }
fileio->handle = hFile; fileio->handle = hFile;
fileio->count = length; fileio->count = length;
@ -756,26 +746,20 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
fileio->queue_apc_on_error = 0; fileio->queue_apc_on_error = 0;
fileio->avail_mode = (flags & FD_FLAG_AVAILABLE); fileio->avail_mode = (flags & FD_FLAG_AVAILABLE);
fileio->event = hEvent; fileio->event = hEvent;
if (needs_close) close( unix_handle );
io_status->Information = 0; io_status->Information = 0;
io_status->u.Status = STATUS_PENDING; io_status->u.Status = STATUS_PENDING;
ret = fileio_queue_async(fileio, io_status, FALSE); status = fileio_queue_async(fileio, io_status, FALSE);
if (ret != STATUS_PENDING) if (status != STATUS_PENDING)
{ {
if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent); if (flags & FD_FLAG_TIMEOUT) NtClose(hEvent);
return ret; goto done;
} }
if (needs_close) close( unix_handle );
if (flags & FD_FLAG_TIMEOUT) if (flags & FD_FLAG_TIMEOUT)
{ {
do while (NtWaitForSingleObject(hEvent, TRUE, NULL) == STATUS_USER_APC) /* nothing */ ;
{
ret = NtWaitForSingleObject(hEvent, TRUE, NULL);
}
while (ret == STATUS_USER_APC && io_status->u.Status == STATUS_PENDING);
NtClose(hEvent); NtClose(hEvent);
if (ret != STATUS_USER_APC)
fileio->queue_apc_on_error = 1;
} }
else else
{ {
@ -783,31 +767,34 @@ NTSTATUS WINAPI NtWriteFile(HANDLE hFile, HANDLE hEvent,
/* let some APC be run, this will write as much data as possible */ /* let some APC be run, this will write as much data as possible */
timeout.u.LowPart = timeout.u.HighPart = 0; timeout.u.LowPart = timeout.u.HighPart = 0;
ret = NtDelayExecution( TRUE, &timeout ); status = NtDelayExecution( TRUE, &timeout );
/* the apc didn't run and therefore the completion routine now /* the apc didn't run and therefore the completion routine now
* needs to be sent errors. * needs to be sent errors.
* Note that there is no race between setting this flag and * Note that there is no race between setting this flag and
* returning errors because apc's are run only during alertable * returning errors because apc's are run only during alertable
* waits */ * waits */
if (ret != STATUS_USER_APC) if (status != STATUS_USER_APC)
fileio->queue_apc_on_error = 1; fileio->queue_apc_on_error = 1;
} }
return io_status->u.Status; return io_status->u.Status;
} }
/* synchronous file write */ /* synchronous file write */
status = STATUS_SUCCESS;
while ((result = write( unix_handle, buffer, length )) == -1) while ((result = write( unix_handle, buffer, length )) == -1)
{ {
if ((errno == EAGAIN) || (errno == EINTR)) continue; if ((errno == EAGAIN) || (errno == EINTR)) continue;
result = 0; result = 0;
if (errno == EFAULT) io_status->u.Status = STATUS_INVALID_USER_BUFFER; if (errno == EFAULT) status = STATUS_INVALID_USER_BUFFER;
else io_status->u.Status = FILE_GetNtStatus(); else status = FILE_GetNtStatus();
break; break;
} }
io_status->Information = result; io_status->Information = result;
done: done:
if (needs_close) close( unix_handle ); if (needs_close) close( unix_handle );
return io_status->u.Status; if (status != STATUS_PENDING) io_status->u.Status = status;
if (apc && !status) apc( apc_user, io_status, 0 );
return status;
} }
/************************************************************************** /**************************************************************************
@ -906,6 +893,8 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
PVOID apc_context, PIO_STATUS_BLOCK io, ULONG code, PVOID apc_context, PIO_STATUS_BLOCK io, ULONG code,
PVOID in_buffer, ULONG in_size, PVOID out_buffer, ULONG out_size) PVOID in_buffer, ULONG in_size, PVOID out_buffer, ULONG out_size)
{ {
NTSTATUS status;
TRACE("(%p,%p,%p,%p,%p,0x%08x,%p,0x%08x,%p,0x%08x)\n", TRACE("(%p,%p,%p,%p,%p,0x%08x,%p,0x%08x,%p,0x%08x)\n",
handle, event, apc, apc_context, io, code, handle, event, apc, apc_context, io, code,
in_buffer, in_size, out_buffer, out_size); in_buffer, in_size, out_buffer, out_size);
@ -915,7 +904,7 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
switch(code) switch(code)
{ {
case FSCTL_DISMOUNT_VOLUME: case FSCTL_DISMOUNT_VOLUME:
io->u.Status = DIR_unmount_device( handle ); status = DIR_unmount_device( handle );
break; break;
case FSCTL_PIPE_LISTEN: case FSCTL_PIPE_LISTEN:
@ -924,8 +913,8 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
if(!event && !apc) if(!event && !apc)
{ {
io->u.Status = NtCreateEvent(&internal_event, EVENT_ALL_ACCESS, NULL, FALSE, FALSE); status = NtCreateEvent(&internal_event, EVENT_ALL_ACCESS, NULL, FALSE, FALSE);
if (io->u.Status != STATUS_SUCCESS) return io->u.Status; if (status != STATUS_SUCCESS) break;
} }
SERVER_START_REQ(connect_named_pipe) SERVER_START_REQ(connect_named_pipe)
{ {
@ -936,13 +925,14 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
req->async.apc = apc; req->async.apc = apc;
req->async.apc_arg = apc_context; req->async.apc_arg = apc_context;
req->async.event = event ? event : internal_event; req->async.event = event ? event : internal_event;
io->u.Status = wine_server_call(req); status = wine_server_call(req);
} }
SERVER_END_REQ; SERVER_END_REQ;
if (internal_event && io->u.Status == STATUS_PENDING) if (internal_event && status == STATUS_PENDING)
{ {
while (NtWaitForSingleObject(internal_event, TRUE, NULL) == STATUS_USER_APC) /*nothing*/ ; while (NtWaitForSingleObject(internal_event, TRUE, NULL) == STATUS_USER_APC) /*nothing*/ ;
status = io->u.Status;
} }
if (internal_event) NtClose(internal_event); if (internal_event) NtClose(internal_event);
} }
@ -955,8 +945,8 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
if(!event && !apc) if(!event && !apc)
{ {
io->u.Status = NtCreateEvent(&internal_event, EVENT_ALL_ACCESS, NULL, FALSE, FALSE); status = NtCreateEvent(&internal_event, EVENT_ALL_ACCESS, NULL, FALSE, FALSE);
if (io->u.Status != STATUS_SUCCESS) return io->u.Status; if (status != STATUS_SUCCESS) break;
} }
SERVER_START_REQ(wait_named_pipe) SERVER_START_REQ(wait_named_pipe)
{ {
@ -970,13 +960,14 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
req->async.apc_arg = apc_context; req->async.apc_arg = apc_context;
req->async.event = event ? event : internal_event; req->async.event = event ? event : internal_event;
wine_server_add_data( req, buff->Name, buff->NameLength ); wine_server_add_data( req, buff->Name, buff->NameLength );
io->u.Status = wine_server_call( req ); status = wine_server_call( req );
} }
SERVER_END_REQ; SERVER_END_REQ;
if (internal_event && io->u.Status == STATUS_PENDING) if (internal_event && status == STATUS_PENDING)
{ {
while (NtWaitForSingleObject(internal_event, TRUE, NULL) == STATUS_USER_APC) /*nothing*/ ; while (NtWaitForSingleObject(internal_event, TRUE, NULL) == STATUS_USER_APC) /*nothing*/ ;
status = io->u.Status;
} }
if (internal_event) NtClose(internal_event); if (internal_event) NtClose(internal_event);
} }
@ -989,18 +980,17 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
if (out_size < FIELD_OFFSET( FILE_PIPE_PEEK_BUFFER, Data )) if (out_size < FIELD_OFFSET( FILE_PIPE_PEEK_BUFFER, Data ))
{ {
io->u.Status = STATUS_INFO_LENGTH_MISMATCH; status = STATUS_INFO_LENGTH_MISMATCH;
break; break;
} }
if ((io->u.Status = server_get_unix_fd( handle, FILE_READ_DATA, &fd, if ((status = server_get_unix_fd( handle, FILE_READ_DATA, &fd, &needs_close, NULL, &flags )))
&needs_close, NULL, &flags )))
break; break;
if (flags & FD_FLAG_RECV_SHUTDOWN) if (flags & FD_FLAG_RECV_SHUTDOWN)
{ {
if (needs_close) close( fd ); if (needs_close) close( fd );
io->u.Status = STATUS_PIPE_DISCONNECTED; status = STATUS_PIPE_DISCONNECTED;
break; break;
} }
@ -1009,7 +999,7 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
{ {
TRACE("FIONREAD failed reason: %s\n",strerror(errno)); TRACE("FIONREAD failed reason: %s\n",strerror(errno));
if (needs_close) close( fd ); if (needs_close) close( fd );
io->u.Status = FILE_GetNtStatus(); status = FILE_GetNtStatus();
break; break;
} }
#endif #endif
@ -1025,7 +1015,7 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
if (ret == -1 || (ret == 1 && (pollfd.revents & (POLLHUP|POLLERR)))) if (ret == -1 || (ret == 1 && (pollfd.revents & (POLLHUP|POLLERR))))
{ {
if (needs_close) close( fd ); if (needs_close) close( fd );
io->u.Status = STATUS_PIPE_BROKEN; status = STATUS_PIPE_BROKEN;
break; break;
} }
} }
@ -1034,7 +1024,7 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
buffer->NumberOfMessages = 0; /* FIXME */ buffer->NumberOfMessages = 0; /* FIXME */
buffer->MessageLength = 0; /* FIXME */ buffer->MessageLength = 0; /* FIXME */
io->Information = FIELD_OFFSET( FILE_PIPE_PEEK_BUFFER, Data ); io->Information = FIELD_OFFSET( FILE_PIPE_PEEK_BUFFER, Data );
io->u.Status = STATUS_SUCCESS; status = STATUS_SUCCESS;
if (avail) if (avail)
{ {
ULONG data_size = out_size - FIELD_OFFSET( FILE_PIPE_PEEK_BUFFER, Data ); ULONG data_size = out_size - FIELD_OFFSET( FILE_PIPE_PEEK_BUFFER, Data );
@ -1052,8 +1042,8 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
SERVER_START_REQ(disconnect_named_pipe) SERVER_START_REQ(disconnect_named_pipe)
{ {
req->handle = handle; req->handle = handle;
io->u.Status = wine_server_call(req); status = wine_server_call(req);
if (!io->u.Status) if (!status)
{ {
int fd = server_remove_fd_from_cache( handle ); int fd = server_remove_fd_from_cache( handle );
if (fd != -1) close( fd ); if (fd != -1) close( fd );
@ -1066,16 +1056,18 @@ NTSTATUS WINAPI NtFsControlFile(HANDLE handle, HANDLE event, PIO_APC_ROUTINE apc
case FSCTL_UNLOCK_VOLUME: case FSCTL_UNLOCK_VOLUME:
FIXME("stub! return success - Unsupported fsctl %x (device=%x access=%x func=%x method=%x)\n", FIXME("stub! return success - Unsupported fsctl %x (device=%x access=%x func=%x method=%x)\n",
code, code >> 16, (code >> 14) & 3, (code >> 2) & 0xfff, code & 3); code, code >> 16, (code >> 14) & 3, (code >> 2) & 0xfff, code & 3);
io->u.Status = STATUS_SUCCESS; status = STATUS_SUCCESS;
break; break;
default: default:
FIXME("Unsupported fsctl %x (device=%x access=%x func=%x method=%x)\n", FIXME("Unsupported fsctl %x (device=%x access=%x func=%x method=%x)\n",
code, code >> 16, (code >> 14) & 3, (code >> 2) & 0xfff, code & 3); code, code >> 16, (code >> 14) & 3, (code >> 2) & 0xfff, code & 3);
io->u.Status = STATUS_NOT_SUPPORTED; status = STATUS_NOT_SUPPORTED;
break; break;
} }
return io->u.Status;
if (status != STATUS_PENDING) io->u.Status = status;
return status;
} }
/****************************************************************************** /******************************************************************************