Win32/ComStreamProxy.cs (108 lines of code) (raw):
// Copyright (c) 2010-2014 SharpDX - Alexandre Mutel
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using System;
using System.IO;
using System.Runtime.InteropServices;
namespace SharpGen.Runtime.Win32
{
[Guid("0000000c-0000-0000-C000-000000000046")]
public class ComStreamProxy : CallbackBase, IStream
{
private Stream sourceStream;
byte[] tempBuffer = new byte[0x1000];
public ComStreamProxy(Stream sourceStream)
{
this.sourceStream = sourceStream;
}
public unsafe uint Read(IntPtr buffer, uint numberOfBytesToRead)
{
uint totalRead = 0;
while (numberOfBytesToRead > 0)
{
uint countRead = (uint)Math.Min(numberOfBytesToRead, tempBuffer.Length);
uint count = (uint)sourceStream.Read(tempBuffer, 0, (int)countRead);
if (count == 0)
return totalRead;
MemoryHelpers.Write(new IntPtr(totalRead + (byte*)buffer), new Span<byte>(tempBuffer), (int)count);
numberOfBytesToRead -= count;
totalRead += count;
}
return totalRead;
}
public unsafe uint Write(IntPtr buffer, uint numberOfBytesToWrite)
{
uint totalWrite = 0;
while (numberOfBytesToWrite > 0)
{
uint countWrite = (uint)Math.Min(numberOfBytesToWrite, tempBuffer.Length);
MemoryHelpers.Read(new IntPtr(totalWrite + (byte*)buffer), new ReadOnlySpan<byte>(tempBuffer), (int)countWrite);
sourceStream.Write(tempBuffer, 0, (int)countWrite);
numberOfBytesToWrite -= countWrite;
totalWrite += countWrite;
}
return totalWrite;
}
public ulong Seek(long offset, SeekOrigin origin)
{
return (ulong)sourceStream.Seek(offset, origin);
}
public void SetSize(ulong newSize)
{
}
public unsafe ulong CopyTo(IStream streamDest, ulong numberOfBytesToCopy, out ulong bytesWritten)
{
bytesWritten = 0;
fixed (void* pBuffer = tempBuffer)
{
while (numberOfBytesToCopy > 0)
{
int countCopy = (int)Math.Min((long)numberOfBytesToCopy, tempBuffer.Length);
int count = sourceStream.Read(tempBuffer, 0, countCopy);
if (count == 0)
break;
streamDest.Write((IntPtr)pBuffer, (uint)count);
numberOfBytesToCopy -= (ulong)count;
bytesWritten += (ulong)count;
}
}
return bytesWritten;
}
public void Commit(CommitFlags commitFlags)
{
sourceStream.Flush();
}
public void Revert()
{
throw new NotImplementedException();
}
public void LockRegion(ulong offset, ulong numberOfBytesToLock, LockType dwLockType)
{
throw new NotImplementedException();
}
public void UnlockRegion(ulong offset, ulong numberOfBytesToLock, LockType dwLockType)
{
throw new NotImplementedException();
}
public StorageStatistics GetStatistics(StorageStatisticsFlags storageStatisticsFlags)
{
long length = sourceStream.Length;
if (length == 0)
length = 0x7fffffff;
return new StorageStatistics
{
Type = 2, // IStream
CbSize = (ulong)length,
GrfLocksSupported = 2, // exclusive
GrfMode = 0x00000002, // read-write
};
}
public IStream Clone()
{
return new ComStreamProxy(sourceStream);
}
protected override void Dispose(bool disposing)
{
sourceStream = null;
base.Dispose(disposing);
}
}
}