[DllImport("kernel32.dll")]
static extern bool ReadFileScatter(IntPtr hFile, FILE_SEGMENT_ELEMENT []
aSegementArray, uint nNumberOfBytesToRead, IntPtr lpReserved,
[In] ref System.Threading.NativeOverlapped lpOverlapped);
or
[DllImport("kernel32.dll", SetLastError=true)]
static extern unsafe int ReadFileScatter(IntPtr hFile,
FILE_SEGMENT_ELEMENT* aSegmentArray, int nNumberOfBytesToRead,
IntPtr lpReserved, NativeOverlapped* lpOverlapped);
[StructLayout(LayoutKind.Explicit, Size = 8)]
internal struct FILE_SEGMENT_ELEMENT
{
[FieldOffset(0)]
public IntPtr Buffer;
[FieldOffset(0)]
public UInt64 Alignment;
}
The documentation for the Scatter/Gather functions states that the buffer addresses used must be page aligned (not just storage aligned). Indeed, passing a managed allocation, such as a byte array, causes error 87 (Invalid Parameter). This function is asynchronous only so an use the AsyncResult and Overlapped pattern established by the FileStream.
It is recommended that VirtualAlloc be used to allocate an unmanaged block of memory for use with this function. The array of file segments can be effeciently constructed without heap allocation using stackalloc. You will need to get the system's page size through a call to GetSystemInfo.
// Prepares the array of file segments given an IntPtr[] ABuffers (only uses ACount of the buffers so that these arrays can be pooled)
// The size of each buffer (ASize) must be a multiple of the system's page size (get using GetSystemInfo)
int LPerBufferPageCount = ASize / FPageSize;
FILE_SEGMENT_ELEMENT* LElements = stackalloc FILE_SEGMENT_ELEMENT[(ACount * LPerBufferPageCount) + 1];
for (int LElementIndex = 0; LElementIndex < ACount; LElementIndex++)
{
for (int LPageIndex = 0; LPageIndex < LPerBufferPageCount; LPageIndex++)
LElements[(LElementIndex * LPerBufferPageCount) + LPageIndex].Buffer =
(IntPtr)((uint)ABuffers[LElementIndex] + (LPageIndex * FPageSize));
}
LElements[(ACount * LPerBufferPageCount)].Buffer = IntPtr.Zero;
No managed API as of the 1.1 Framework.