AxMemoryStream.cs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Runtime.InteropServices;
  5. using System.IO;
  6. namespace Biff8Excel.COM
  7. {
  8. [ClassInterface(ClassInterfaceType.AutoDispatch)]
  9. public class AxMemoryStream : MemoryStream, IStream
  10. {
  11. void IStream.Clone(out IStream clone)
  12. {
  13. clone = this.MemberwiseClone() as IStream;
  14. }
  15. void IStream.Commit(int flags)
  16. {
  17. throw new NotImplementedException("AxMemoryStream is not transactional");
  18. }
  19. void IStream.CopyTo(IStream destination, long count, IntPtr pcbRead, IntPtr pcbWritten)
  20. {
  21. /**///////////
  22. // Copy the lot using 4k chunks
  23. /**///////////
  24. byte[] _buffer = new byte[4096];
  25. int _cbRead = 0;
  26. int _cbWritten = 0;
  27. IntPtr p = IntPtr.Zero;
  28. while (count > 0)
  29. {
  30. int _chunk = (int)Math.Min(count, _buffer.Length);
  31. int _chunkRead = this.Read(_buffer, _cbRead, _chunk);
  32. destination.Write(_buffer, _chunk, ref p);
  33. _cbRead += _chunkRead;
  34. _cbWritten += _chunkRead;
  35. }
  36. /**///////////
  37. // Update the counts, if they were provided
  38. /**///////////
  39. if (pcbRead != IntPtr.Zero)
  40. {
  41. Marshal.WriteInt64(pcbRead, _cbRead);
  42. }
  43. if (pcbWritten != IntPtr.Zero)
  44. {
  45. Marshal.WriteInt64(pcbWritten, _cbWritten);
  46. }
  47. }
  48. void IStream.LockRegion(long offset, long count, int lockType)
  49. {
  50. throw new NotImplementedException("AxMemoryStream does not support locking");
  51. }
  52. void IStream.Read(byte[] buffer, int count,ref IntPtr pcbRead)
  53. {
  54. long _cbRead = 0;
  55. int offset = 0;
  56. int cnt = int.MaxValue;
  57. byte[] tmp = new byte[cnt];
  58. while (count > 0)
  59. {
  60. _cbRead += this.Read(tmp, offset, cnt);
  61. tmp.CopyTo(buffer, offset);
  62. offset += cnt;
  63. count -= cnt;
  64. if ((count - cnt) < int.MaxValue)
  65. {
  66. _cbRead += this.Read(tmp, offset, (int)(count) );
  67. tmp.CopyTo(buffer, offset);
  68. break;
  69. }
  70. }
  71. if (pcbRead != IntPtr.Zero)
  72. {
  73. //Marshal.WriteInt32(pcbRead, _cbRead);
  74. Marshal.WriteInt64(pcbRead, _cbRead);
  75. }
  76. //throw new NotImplementedException("AxMemoryStream does not support Read");
  77. }
  78. void IStream.Revert()
  79. {
  80. throw new NotImplementedException("AxMemoryStream is not transactional");
  81. }
  82. void IStream.Seek(long offset, int origin,ref IntPtr pcbPos)
  83. {
  84. long _position = this.Seek(offset, (SeekOrigin)origin);
  85. if (pcbPos != IntPtr.Zero)
  86. {
  87. Marshal.WriteInt64(pcbPos, _position);
  88. }
  89. }
  90. void IStream.SetSize(long newSize)
  91. {
  92. this.SetLength(newSize);
  93. }
  94. void IStream.Stat(out STATSTG stat, int flags)
  95. {
  96. stat = new STATSTG();
  97. stat.cbSize = Marshal.SizeOf(stat);
  98. stat.grfLocksSupported = 0;
  99. }
  100. void IStream.UnlockRegion(long offset, long count, int lockType)
  101. {
  102. throw new NotImplementedException("AxMemoryStream does not support locking");
  103. }
  104. void IStream.Write(byte[] buffer, int count,ref IntPtr pcbWritten)
  105. {
  106. this.Write(buffer, 0, count);
  107. if (pcbWritten != IntPtr.Zero)
  108. {
  109. Marshal.WriteInt32(pcbWritten, count);
  110. }
  111. }
  112. }
  113. }