KPageTableHostMapped.cs 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. using Ryujinx.HLE.HOS.Kernel.Common;
  2. using Ryujinx.Memory;
  3. using Ryujinx.Memory.Range;
  4. using System;
  5. using System.Collections.Generic;
  6. using System.Linq;
  7. namespace Ryujinx.HLE.HOS.Kernel.Memory
  8. {
  9. class KPageTableHostMapped : KPageTableBase
  10. {
  11. private const int CopyChunckSize = 0x100000;
  12. private readonly IVirtualMemoryManager _cpuMemory;
  13. public override bool SupportsMemoryAliasing => false;
  14. public KPageTableHostMapped(KernelContext context, IVirtualMemoryManager cpuMemory) : base(context)
  15. {
  16. _cpuMemory = cpuMemory;
  17. }
  18. /// <inheritdoc/>
  19. protected override IEnumerable<HostMemoryRange> GetPhysicalRegions(ulong va, ulong size)
  20. {
  21. return _cpuMemory.GetPhysicalRegions(va, size);
  22. }
  23. /// <inheritdoc/>
  24. protected override ReadOnlySpan<byte> GetSpan(ulong va, int size)
  25. {
  26. return _cpuMemory.GetSpan(va, size);
  27. }
  28. /// <inheritdoc/>
  29. protected override KernelResult MapMemory(ulong src, ulong dst, ulong pagesCount, KMemoryPermission oldSrcPermission, KMemoryPermission newDstPermission)
  30. {
  31. ulong size = pagesCount * PageSize;
  32. _cpuMemory.Map(dst, 0, size);
  33. ulong currentSize = size;
  34. while (currentSize > 0)
  35. {
  36. ulong copySize = Math.Min(currentSize, CopyChunckSize);
  37. _cpuMemory.Write(dst, _cpuMemory.GetSpan(src, (int)copySize));
  38. currentSize -= copySize;
  39. }
  40. return KernelResult.Success;
  41. }
  42. /// <inheritdoc/>
  43. protected override KernelResult UnmapMemory(ulong dst, ulong src, ulong pagesCount, KMemoryPermission oldDstPermission, KMemoryPermission newSrcPermission)
  44. {
  45. ulong size = pagesCount * PageSize;
  46. // TODO: Validation.
  47. ulong currentSize = size;
  48. while (currentSize > 0)
  49. {
  50. ulong copySize = Math.Min(currentSize, CopyChunckSize);
  51. _cpuMemory.Write(src, _cpuMemory.GetSpan(dst, (int)copySize));
  52. currentSize -= copySize;
  53. }
  54. _cpuMemory.Unmap(dst, size);
  55. return KernelResult.Success;
  56. }
  57. /// <inheritdoc/>
  58. protected override KernelResult MapPages(ulong dstVa, ulong pagesCount, ulong srcPa, KMemoryPermission permission)
  59. {
  60. _cpuMemory.Map(dstVa, 0, pagesCount * PageSize);
  61. return KernelResult.Success;
  62. }
  63. /// <inheritdoc/>
  64. protected override KernelResult MapPages(ulong address, KPageList pageList, KMemoryPermission permission)
  65. {
  66. _cpuMemory.Map(address, 0, pageList.GetPagesCount() * PageSize);
  67. return KernelResult.Success;
  68. }
  69. /// <inheritdoc/>
  70. protected override KernelResult MapPages(ulong address, IEnumerable<HostMemoryRange> ranges, KMemoryPermission permission)
  71. {
  72. throw new NotSupportedException();
  73. }
  74. /// <inheritdoc/>
  75. protected override KernelResult Unmap(ulong address, ulong pagesCount)
  76. {
  77. _cpuMemory.Unmap(address, pagesCount * PageSize);
  78. return KernelResult.Success;
  79. }
  80. /// <inheritdoc/>
  81. protected override KernelResult Reprotect(ulong address, ulong pagesCount, KMemoryPermission permission)
  82. {
  83. // TODO.
  84. return KernelResult.Success;
  85. }
  86. /// <inheritdoc/>
  87. protected override KernelResult ReprotectWithAttributes(ulong address, ulong pagesCount, KMemoryPermission permission)
  88. {
  89. // TODO.
  90. return KernelResult.Success;
  91. }
  92. /// <inheritdoc/>
  93. protected override void SignalMemoryTracking(ulong va, ulong size, bool write)
  94. {
  95. _cpuMemory.SignalMemoryTracking(va, size, write);
  96. }
  97. /// <inheritdoc/>
  98. protected override void Write(ulong va, ReadOnlySpan<byte> data)
  99. {
  100. _cpuMemory.Write(va, data);
  101. }
  102. }
  103. }