MultiRegionHandle.cs 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. using System;
  2. namespace Ryujinx.Memory.Tracking
  3. {
  4. /// <summary>
  5. /// A region handle that tracks a large region using many smaller handles, to provide
  6. /// granular tracking that can be used to track partial updates.
  7. /// </summary>
  8. public class MultiRegionHandle : IMultiRegionHandle
  9. {
  10. /// <summary>
  11. /// A list of region handles for each granularity sized chunk of the whole region.
  12. /// </summary>
  13. private readonly RegionHandle[] _handles;
  14. private readonly ulong Address;
  15. private readonly ulong Granularity;
  16. private readonly ulong Size;
  17. public bool Dirty { get; private set; } = true;
  18. internal MultiRegionHandle(MemoryTracking tracking, ulong address, ulong size, ulong granularity)
  19. {
  20. _handles = new RegionHandle[size / granularity];
  21. Granularity = granularity;
  22. for (int i = 0; i < _handles.Length; i++)
  23. {
  24. RegionHandle handle = tracking.BeginTracking(address + (ulong)i * granularity, granularity);
  25. handle.Parent = this;
  26. _handles[i] = handle;
  27. }
  28. Address = address;
  29. Size = size;
  30. }
  31. public void SignalWrite()
  32. {
  33. Dirty = true;
  34. }
  35. public void QueryModified(Action<ulong, ulong> modifiedAction)
  36. {
  37. if (!Dirty)
  38. {
  39. return;
  40. }
  41. Dirty = false;
  42. QueryModified(Address, Size, modifiedAction);
  43. }
  44. public void QueryModified(ulong address, ulong size, Action<ulong, ulong> modifiedAction)
  45. {
  46. int startHandle = (int)((address - Address) / Granularity);
  47. int lastHandle = (int)((address + (size - 1) - Address) / Granularity);
  48. ulong rgStart = _handles[startHandle].Address;
  49. ulong rgSize = 0;
  50. for (int i = startHandle; i <= lastHandle; i++)
  51. {
  52. RegionHandle handle = _handles[i];
  53. if (handle.Dirty)
  54. {
  55. rgSize += handle.Size;
  56. handle.Reprotect();
  57. }
  58. else
  59. {
  60. // Submit the region scanned so far as dirty
  61. if (rgSize != 0)
  62. {
  63. modifiedAction(rgStart, rgSize);
  64. rgSize = 0;
  65. }
  66. rgStart = handle.EndAddress;
  67. }
  68. }
  69. if (rgSize != 0)
  70. {
  71. modifiedAction(rgStart, rgSize);
  72. }
  73. }
  74. public void QueryModified(ulong address, ulong size, Action<ulong, ulong> modifiedAction, int sequenceNumber)
  75. {
  76. int startHandle = (int)((address - Address) / Granularity);
  77. int lastHandle = (int)((address + (size - 1) - Address) / Granularity);
  78. ulong rgStart = _handles[startHandle].Address;
  79. ulong rgSize = 0;
  80. for (int i = startHandle; i <= lastHandle; i++)
  81. {
  82. RegionHandle handle = _handles[i];
  83. if (handle.Dirty && sequenceNumber != handle.SequenceNumber)
  84. {
  85. rgSize += handle.Size;
  86. handle.Reprotect();
  87. }
  88. else
  89. {
  90. // Submit the region scanned so far as dirty
  91. if (rgSize != 0)
  92. {
  93. modifiedAction(rgStart, rgSize);
  94. rgSize = 0;
  95. }
  96. rgStart = handle.EndAddress;
  97. }
  98. handle.SequenceNumber = sequenceNumber;
  99. }
  100. if (rgSize != 0)
  101. {
  102. modifiedAction(rgStart, rgSize);
  103. }
  104. }
  105. public void Dispose()
  106. {
  107. foreach (var handle in _handles)
  108. {
  109. handle.Dispose();
  110. }
  111. }
  112. }
  113. }