| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129 |
- using Ryujinx.Common;
- using Ryujinx.HLE.HOS.Kernel.Common;
- using Ryujinx.HLE.HOS.Kernel.Process;
- using System;
- namespace Ryujinx.HLE.HOS.Kernel.Memory
- {
- class KTransferMemory : KAutoObject
- {
- private KProcess _creator;
- // TODO: Remove when we no longer need to read it from the owner directly.
- public KProcess Creator => _creator;
- private readonly KPageList _pageList;
- public ulong Address { get; private set; }
- public ulong Size { get; private set; }
- public KMemoryPermission Permission { get; private set; }
- private bool _hasBeenInitialized;
- private bool _isMapped;
- public KTransferMemory(KernelContext context) : base(context)
- {
- _pageList = new KPageList();
- }
- public KTransferMemory(KernelContext context, SharedMemoryStorage storage) : base(context)
- {
- _pageList = storage.GetPageList();
- Permission = KMemoryPermission.ReadAndWrite;
- _hasBeenInitialized = true;
- _isMapped = false;
- }
- public KernelResult Initialize(ulong address, ulong size, KMemoryPermission permission)
- {
- KProcess creator = KernelStatic.GetCurrentProcess();
- _creator = creator;
- KernelResult result = creator.MemoryManager.BorrowTransferMemory(_pageList, address, size, permission);
- if (result != KernelResult.Success)
- {
- return result;
- }
- creator.IncrementReferenceCount();
- Permission = permission;
- Address = address;
- Size = size;
- _hasBeenInitialized = true;
- _isMapped = false;
- return result;
- }
- public KernelResult MapIntoProcess(
- KPageTableBase memoryManager,
- ulong address,
- ulong size,
- KProcess process,
- KMemoryPermission permission)
- {
- if (_pageList.GetPagesCount() != BitUtils.DivRoundUp(size, KPageTableBase.PageSize))
- {
- return KernelResult.InvalidSize;
- }
- if (permission != Permission || _isMapped)
- {
- return KernelResult.InvalidState;
- }
- MemoryState state = Permission == KMemoryPermission.None ? MemoryState.TransferMemoryIsolated : MemoryState.TransferMemory;
- KernelResult result = memoryManager.MapPages(address, _pageList, state, KMemoryPermission.ReadAndWrite);
- if (result == KernelResult.Success)
- {
- _isMapped = true;
- }
- return result;
- }
- public KernelResult UnmapFromProcess(
- KPageTableBase memoryManager,
- ulong address,
- ulong size,
- KProcess process)
- {
- if (_pageList.GetPagesCount() != BitUtils.DivRoundUp(size, KPageTableBase.PageSize))
- {
- return KernelResult.InvalidSize;
- }
- MemoryState state = Permission == KMemoryPermission.None ? MemoryState.TransferMemoryIsolated : MemoryState.TransferMemory;
- KernelResult result = memoryManager.UnmapPages(address, _pageList, state);
- if (result == KernelResult.Success)
- {
- _isMapped = false;
- }
- return result;
- }
- protected override void Destroy()
- {
- if (_hasBeenInitialized)
- {
- if (!_isMapped && _creator.MemoryManager.UnborrowTransferMemory(Address, Size, _pageList) != KernelResult.Success)
- {
- throw new InvalidOperationException("Unexpected failure restoring transfer memory attributes.");
- }
- _creator.ResourceLimit?.Release(LimitableResource.TransferMemory, 1);
- _creator.DecrementReferenceCount();
- }
- }
- }
- }
|