| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168 |
- using Ryujinx.Common;
- using Ryujinx.HLE.HOS.Kernel.Common;
- using Ryujinx.HLE.HOS.Kernel.Process;
- using System;
- using System.Diagnostics;
- namespace Ryujinx.HLE.HOS.Kernel.Memory
- {
- class KCodeMemory : KAutoObject
- {
- public KProcess Owner { get; private set; }
- private readonly KPageList _pageList;
- private readonly object _lock;
- private ulong _address;
- private bool _isOwnerMapped;
- private bool _isMapped;
- public KCodeMemory(KernelContext context) : base(context)
- {
- _pageList = new KPageList();
- _lock = new object();
- }
- public KernelResult Initialize(ulong address, ulong size)
- {
- Owner = KernelStatic.GetCurrentProcess();
- KernelResult result = Owner.MemoryManager.BorrowCodeMemory(_pageList, address, size);
- if (result != KernelResult.Success)
- {
- return result;
- }
- Owner.CpuMemory.Fill(address, size, 0xff);
- Owner.IncrementReferenceCount();
- _address = address;
- _isMapped = false;
- _isOwnerMapped = false;
- return KernelResult.Success;
- }
- public KernelResult Map(ulong address, ulong size, KMemoryPermission perm)
- {
- if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, (ulong)KPageTableBase.PageSize))
- {
- return KernelResult.InvalidSize;
- }
- lock (_lock)
- {
- if (_isMapped)
- {
- return KernelResult.InvalidState;
- }
- KProcess process = KernelStatic.GetCurrentProcess();
- KernelResult result = process.MemoryManager.MapPages(address, _pageList, MemoryState.CodeWritable, KMemoryPermission.ReadAndWrite);
- if (result != KernelResult.Success)
- {
- return result;
- }
- _isMapped = true;
- }
- return KernelResult.Success;
- }
- public KernelResult MapToOwner(ulong address, ulong size, KMemoryPermission permission)
- {
- if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, (ulong)KPageTableBase.PageSize))
- {
- return KernelResult.InvalidSize;
- }
- lock (_lock)
- {
- if (_isOwnerMapped)
- {
- return KernelResult.InvalidState;
- }
- Debug.Assert(permission == KMemoryPermission.Read || permission == KMemoryPermission.ReadAndExecute);
- KernelResult result = Owner.MemoryManager.MapPages(address, _pageList, MemoryState.CodeReadOnly, permission);
- if (result != KernelResult.Success)
- {
- return result;
- }
- _isOwnerMapped = true;
- }
- return KernelResult.Success;
- }
- public KernelResult Unmap(ulong address, ulong size)
- {
- if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, (ulong)KPageTableBase.PageSize))
- {
- return KernelResult.InvalidSize;
- }
- lock (_lock)
- {
- KProcess process = KernelStatic.GetCurrentProcess();
- KernelResult result = process.MemoryManager.UnmapPages(address, _pageList, MemoryState.CodeWritable);
- if (result != KernelResult.Success)
- {
- return result;
- }
- Debug.Assert(_isMapped);
- _isMapped = false;
- }
- return KernelResult.Success;
- }
- public KernelResult UnmapFromOwner(ulong address, ulong size)
- {
- if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, KPageTableBase.PageSize))
- {
- return KernelResult.InvalidSize;
- }
- lock (_lock)
- {
- KernelResult result = Owner.MemoryManager.UnmapPages(address, _pageList, MemoryState.CodeReadOnly);
- if (result != KernelResult.Success)
- {
- return result;
- }
- Debug.Assert(_isOwnerMapped);
- _isOwnerMapped = false;
- }
- return KernelResult.Success;
- }
- protected override void Destroy()
- {
- if (!_isMapped && !_isOwnerMapped)
- {
- ulong size = _pageList.GetPagesCount() * KPageTableBase.PageSize;
- if (Owner.MemoryManager.UnborrowCodeMemory(_address, size, _pageList) != KernelResult.Success)
- {
- throw new InvalidOperationException("Unexpected failure restoring transfer memory attributes.");
- }
- }
- Owner.DecrementReferenceCount();
- }
- }
- }
|