RamStorage.cs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. using System;
  2. using System.Runtime.InteropServices;
  3. namespace SvdCli.Storage.Implements
  4. {
  5. public class UnmanagedGigabyteRamDisk : ISvdStorage
  6. {
  7. private readonly IntPtr[] _blocks;
  8. private bool _isInit;
  9. public string ProductId { get; } = "Svd RamDisk";
  10. public string ProductVersion { get; } = "1.0";
  11. public uint BlockSize { get; } = 512;
  12. public ulong BlockCount { get; }
  13. public uint MaxTransferLength { get; } = 64 * CapacityUnits.KiloByte;
  14. public bool SupportTrim { get; } = false;
  15. public bool WriteProtect { get; set; }
  16. public event EventHandler Mounted;
  17. public UnmanagedGigabyteRamDisk(int sizeInGigabyte)
  18. {
  19. _blocks = new IntPtr[sizeInGigabyte];
  20. var size = (ulong)sizeInGigabyte * CapacityUnits.Gigabyte;
  21. BlockCount = size / BlockSize;
  22. }
  23. public void Init()
  24. {
  25. if (_isInit) return;
  26. try
  27. {
  28. for (var i = 0; i < _blocks.Length; i++)
  29. {
  30. _blocks[i] = Marshal.AllocHGlobal(CapacityUnits.Gigabyte);
  31. }
  32. _isInit = true;
  33. }
  34. catch (OutOfMemoryException)
  35. {
  36. Exit();
  37. throw;
  38. }
  39. }
  40. public void Exit()
  41. {
  42. foreach (var block in _blocks)
  43. {
  44. if (block != IntPtr.Zero) Marshal.FreeHGlobal(block);
  45. }
  46. GC.Collect();
  47. GC.WaitForPendingFinalizers();
  48. _isInit = false;
  49. }
  50. private void ReadInternal(byte[] buffer, int offset, int count, int gigabyteIndex, int gigabyteOffset)
  51. {
  52. Marshal.Copy(_blocks[gigabyteIndex] + gigabyteOffset, buffer, offset, count);
  53. }
  54. private void WriteInternal(byte[] buffer, int offset, int count, int gigabyteIndex, int gigabyteOffset)
  55. {
  56. Marshal.Copy(buffer, offset, _blocks[gigabyteIndex] + gigabyteOffset, count);
  57. }
  58. public void ReadBlocks(byte[] buffer, ulong blockIndex, uint blockCount)
  59. {
  60. var rawOffset = blockIndex * BlockSize;
  61. var rawSize = blockCount * BlockSize;
  62. for (var i = 0; i < rawSize;)
  63. {
  64. var gigabyteIndex = (int)(rawOffset / CapacityUnits.Gigabyte);
  65. var gigabyteOffset = (int)(rawOffset % CapacityUnits.Gigabyte);
  66. var bytesToRead = CapacityUnits.Gigabyte - gigabyteOffset;
  67. if (i + bytesToRead > rawSize)
  68. {
  69. bytesToRead = (int)(rawSize - i);
  70. }
  71. ReadInternal(buffer, i, bytesToRead, gigabyteIndex, gigabyteOffset);
  72. i += bytesToRead;
  73. rawOffset += (ulong)bytesToRead;
  74. }
  75. }
  76. public void WriteBlocks(byte[] buffer, ulong blockIndex, uint blockCount)
  77. {
  78. var rawOffset = blockIndex * BlockSize;
  79. var rawSize = (int)(blockCount * BlockSize);
  80. for (var i = 0; i < rawSize;)
  81. {
  82. var gigabyteIndex = (int)(rawOffset / CapacityUnits.Gigabyte);
  83. var gigabyteOffset = (int)(rawOffset % CapacityUnits.Gigabyte);
  84. var bytesToWrite = CapacityUnits.Gigabyte - gigabyteOffset;
  85. if (i + bytesToWrite > rawSize)
  86. {
  87. bytesToWrite = rawSize - i;
  88. }
  89. WriteInternal(buffer, i, bytesToWrite, gigabyteIndex, gigabyteOffset);
  90. i += bytesToWrite;
  91. rawOffset += (ulong)bytesToWrite;
  92. }
  93. }
  94. public int Read(byte[] buffer, long offset, int index, int count)
  95. {
  96. for (var i = 0; i < count;)
  97. {
  98. var gigabyteIndex = (int)(offset / CapacityUnits.Gigabyte);
  99. var gigabyteOffset = (int)(offset % CapacityUnits.Gigabyte);
  100. var bytesToRead = CapacityUnits.Gigabyte - gigabyteOffset;
  101. if (i + bytesToRead > count)
  102. {
  103. bytesToRead = (count - i);
  104. }
  105. ReadInternal(buffer, i, bytesToRead, gigabyteIndex, gigabyteOffset);
  106. i += bytesToRead;
  107. offset += bytesToRead;
  108. }
  109. return count;
  110. }
  111. public void Write(byte[] buffer, long offset, int index, int count)
  112. {
  113. for (var i = 0; i < count;)
  114. {
  115. var gigabyteIndex = (int)(offset / CapacityUnits.Gigabyte);
  116. var gigabyteOffset = (int)(offset % CapacityUnits.Gigabyte);
  117. var bytesToWrite = CapacityUnits.Gigabyte - gigabyteOffset;
  118. if (i + bytesToWrite > count)
  119. {
  120. bytesToWrite = count - i;
  121. }
  122. WriteInternal(buffer, i, bytesToWrite, gigabyteIndex, gigabyteOffset);
  123. i += bytesToWrite;
  124. offset += bytesToWrite;
  125. }
  126. }
  127. public void Flush()
  128. {
  129. }
  130. public void Trim(params TrimDescriptor[] descriptors)
  131. {
  132. }
  133. public void FireMountedEvent()
  134. {
  135. Mounted?.Invoke(this, EventArgs.Empty);
  136. }
  137. }
  138. }