Commit 97df230a authored by Vladislav Vaintroub's avatar Vladislav Vaintroub

MDEV-14115 : Do not use lpNumberOfBytesRead/Written params in

ReadFile/WriteFile operations.

Innodb opens files with FILE_FLAG_OVERLAPPED. lpNumberOfBytesRead/Written
are documented to be potentially inaccurate in this case,
(possibly even if async operations complete synchronously?)

The fix is to always pass NULL for the correspondng parameters,
as recommended by  MSDN. Read the actual counts with
GetQueuedCompletionStatus() or GetOverlappedResult().
parent 067f8396
...@@ -775,9 +775,9 @@ os_win32_device_io_control( ...@@ -775,9 +775,9 @@ os_win32_device_io_control(
OVERLAPPED overlapped = { 0 }; OVERLAPPED overlapped = { 0 };
overlapped.hEvent = win_get_syncio_event(); overlapped.hEvent = win_get_syncio_event();
BOOL result = DeviceIoControl(handle, code, inbuf, inbuf_size, outbuf, BOOL result = DeviceIoControl(handle, code, inbuf, inbuf_size, outbuf,
outbuf_size, bytes_returned, &overlapped); outbuf_size, NULL, &overlapped);
if (!result && (GetLastError() == ERROR_IO_PENDING)) { if (result || (GetLastError() == ERROR_IO_PENDING)) {
/* Wait for async io to complete */ /* Wait for async io to complete */
result = GetOverlappedResult(handle, &overlapped, bytes_returned, TRUE); result = GetOverlappedResult(handle, &overlapped, bytes_returned, TRUE);
} }
...@@ -3451,14 +3451,14 @@ SyncFileIO::execute(const IORequest& request) ...@@ -3451,14 +3451,14 @@ SyncFileIO::execute(const IORequest& request)
if (request.is_read()) { if (request.is_read()) {
ret = ReadFile(m_fh, m_buf, ret = ReadFile(m_fh, m_buf,
static_cast<DWORD>(m_n), &n_bytes, &seek); static_cast<DWORD>(m_n), NULL, &seek);
} else { } else {
ut_ad(request.is_write()); ut_ad(request.is_write());
ret = WriteFile(m_fh, m_buf, ret = WriteFile(m_fh, m_buf,
static_cast<DWORD>(m_n), &n_bytes, &seek); static_cast<DWORD>(m_n), NULL, &seek);
} }
if (!ret && (GetLastError() == ERROR_IO_PENDING)) { if (ret || (GetLastError() == ERROR_IO_PENDING)) {
/* Wait for async io to complete */ /* Wait for async io to complete */
ret = GetOverlappedResult(m_fh, &seek, &n_bytes, TRUE); ret = GetOverlappedResult(m_fh, &seek, &n_bytes, TRUE);
} }
...@@ -3478,17 +3478,17 @@ SyncFileIO::execute(Slot* slot) ...@@ -3478,17 +3478,17 @@ SyncFileIO::execute(Slot* slot)
ret = ReadFile( ret = ReadFile(
slot->file, slot->ptr, slot->len, slot->file, slot->ptr, slot->len,
&slot->n_bytes, &slot->control); NULL, &slot->control);
} else { } else {
ut_ad(slot->type.is_write()); ut_ad(slot->type.is_write());
ret = WriteFile( ret = WriteFile(
slot->file, slot->ptr, slot->len, slot->file, slot->ptr, slot->len,
&slot->n_bytes, &slot->control); NULL, &slot->control);
} }
if (!ret && (GetLastError() == ERROR_IO_PENDING)) { if (ret || (GetLastError() == ERROR_IO_PENDING)) {
/* Wait for async io to complete */ /* Wait for async io to complete */
ret = GetOverlappedResult(slot->file, &slot->control, &slot->n_bytes, TRUE); ret = GetOverlappedResult(slot->file, &slot->control, &slot->n_bytes, TRUE);
} }
...@@ -6725,7 +6725,7 @@ os_aio_func( ...@@ -6725,7 +6725,7 @@ os_aio_func(
#ifdef WIN_ASYNC_IO #ifdef WIN_ASYNC_IO
ret = ReadFile( ret = ReadFile(
file, slot->ptr, slot->len, file, slot->ptr, slot->len,
&slot->n_bytes, &slot->control); NULL, &slot->control);
#elif defined(LINUX_NATIVE_AIO) #elif defined(LINUX_NATIVE_AIO)
if (!array->linux_dispatch(slot)) { if (!array->linux_dispatch(slot)) {
goto err_exit; goto err_exit;
...@@ -6743,7 +6743,7 @@ os_aio_func( ...@@ -6743,7 +6743,7 @@ os_aio_func(
#ifdef WIN_ASYNC_IO #ifdef WIN_ASYNC_IO
ret = WriteFile( ret = WriteFile(
file, slot->ptr, slot->len, file, slot->ptr, slot->len,
&slot->n_bytes, &slot->control); NULL, &slot->control);
#elif defined(LINUX_NATIVE_AIO) #elif defined(LINUX_NATIVE_AIO)
if (!array->linux_dispatch(slot)) { if (!array->linux_dispatch(slot)) {
goto err_exit; goto err_exit;
...@@ -6759,8 +6759,7 @@ os_aio_func( ...@@ -6759,8 +6759,7 @@ os_aio_func(
} }
#ifdef WIN_ASYNC_IO #ifdef WIN_ASYNC_IO
if ((ret && slot->len == slot->n_bytes) if (ret || (GetLastError() == ERROR_IO_PENDING)) {
|| (!ret && GetLastError() == ERROR_IO_PENDING)) {
/* aio completed or was queued successfully! */ /* aio completed or was queued successfully! */
return(DB_SUCCESS); return(DB_SUCCESS);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment