KCodeMemory.cs 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. using Ryujinx.Common;
  2. using Ryujinx.HLE.HOS.Kernel.Common;
  3. using Ryujinx.HLE.HOS.Kernel.Process;
  4. using System;
  5. using System.Diagnostics;
  6. namespace Ryujinx.HLE.HOS.Kernel.Memory
  7. {
  8. class KCodeMemory : KAutoObject
  9. {
  10. public KProcess Owner { get; private set; }
  11. private readonly KPageList _pageList;
  12. private readonly object _lock;
  13. private ulong _address;
  14. private bool _isOwnerMapped;
  15. private bool _isMapped;
  16. public KCodeMemory(KernelContext context) : base(context)
  17. {
  18. _pageList = new KPageList();
  19. _lock = new object();
  20. }
  21. public KernelResult Initialize(ulong address, ulong size)
  22. {
  23. Owner = KernelStatic.GetCurrentProcess();
  24. KernelResult result = Owner.MemoryManager.BorrowCodeMemory(_pageList, address, size);
  25. if (result != KernelResult.Success)
  26. {
  27. return result;
  28. }
  29. Owner.CpuMemory.Fill(address, size, 0xff);
  30. Owner.IncrementReferenceCount();
  31. _address = address;
  32. _isMapped = false;
  33. _isOwnerMapped = false;
  34. return KernelResult.Success;
  35. }
  36. public KernelResult Map(ulong address, ulong size, KMemoryPermission perm)
  37. {
  38. if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, (ulong)KPageTableBase.PageSize))
  39. {
  40. return KernelResult.InvalidSize;
  41. }
  42. lock (_lock)
  43. {
  44. if (_isMapped)
  45. {
  46. return KernelResult.InvalidState;
  47. }
  48. KProcess process = KernelStatic.GetCurrentProcess();
  49. KernelResult result = process.MemoryManager.MapPages(address, _pageList, MemoryState.CodeWritable, KMemoryPermission.ReadAndWrite);
  50. if (result != KernelResult.Success)
  51. {
  52. return result;
  53. }
  54. _isMapped = true;
  55. }
  56. return KernelResult.Success;
  57. }
  58. public KernelResult MapToOwner(ulong address, ulong size, KMemoryPermission permission)
  59. {
  60. if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, (ulong)KPageTableBase.PageSize))
  61. {
  62. return KernelResult.InvalidSize;
  63. }
  64. lock (_lock)
  65. {
  66. if (_isOwnerMapped)
  67. {
  68. return KernelResult.InvalidState;
  69. }
  70. Debug.Assert(permission == KMemoryPermission.Read || permission == KMemoryPermission.ReadAndExecute);
  71. KernelResult result = Owner.MemoryManager.MapPages(address, _pageList, MemoryState.CodeReadOnly, permission);
  72. if (result != KernelResult.Success)
  73. {
  74. return result;
  75. }
  76. _isOwnerMapped = true;
  77. }
  78. return KernelResult.Success;
  79. }
  80. public KernelResult Unmap(ulong address, ulong size)
  81. {
  82. if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, (ulong)KPageTableBase.PageSize))
  83. {
  84. return KernelResult.InvalidSize;
  85. }
  86. lock (_lock)
  87. {
  88. KProcess process = KernelStatic.GetCurrentProcess();
  89. KernelResult result = process.MemoryManager.UnmapPages(address, _pageList, MemoryState.CodeWritable);
  90. if (result != KernelResult.Success)
  91. {
  92. return result;
  93. }
  94. Debug.Assert(_isMapped);
  95. _isMapped = false;
  96. }
  97. return KernelResult.Success;
  98. }
  99. public KernelResult UnmapFromOwner(ulong address, ulong size)
  100. {
  101. if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, KPageTableBase.PageSize))
  102. {
  103. return KernelResult.InvalidSize;
  104. }
  105. lock (_lock)
  106. {
  107. KernelResult result = Owner.MemoryManager.UnmapPages(address, _pageList, MemoryState.CodeReadOnly);
  108. if (result != KernelResult.Success)
  109. {
  110. return result;
  111. }
  112. Debug.Assert(_isOwnerMapped);
  113. _isOwnerMapped = false;
  114. }
  115. return KernelResult.Success;
  116. }
  117. protected override void Destroy()
  118. {
  119. if (!_isMapped && !_isOwnerMapped)
  120. {
  121. ulong size = _pageList.GetPagesCount() * KPageTableBase.PageSize;
  122. if (Owner.MemoryManager.UnborrowCodeMemory(_address, size, _pageList) != KernelResult.Success)
  123. {
  124. throw new InvalidOperationException("Unexpected failure restoring transfer memory attributes.");
  125. }
  126. }
  127. Owner.DecrementReferenceCount();
  128. }
  129. }
  130. }