aboutsummaryrefslogblamecommitdiff
path: root/src/Ryujinx.HLE/HOS/Kernel/Memory/KCodeMemory.cs
blob: c725501b05c3156a216e0d405435b899e3437576 (plain) (tree)
1
2
3
4

                                     
                             


















                                                                 
                                                           

                                                     
                                                                                           
 
                                         









                                                      
                                  
         
                                                                            
         
                                                                                                              











                                                                    
                                                                                                                                             
 
                                             





                                  
                                  
         
                                                                                         
         
                                                                                                              











                                                                                                                     
                                                                                                                       
 
                                             





                                      
                                  
         
                                                      
         
                                                                                                              






                                                                    
                                                                                                               
 
                                             







                                        
                                  
         
                                                               
         
                                                                                                       




                                                
                                                                                                             
 
                                             







                                             
                                  






                                                                                 
                                                                                                        






                                                                                                                    
 
using Ryujinx.Common;
using Ryujinx.HLE.HOS.Kernel.Common;
using Ryujinx.HLE.HOS.Kernel.Process;
using Ryujinx.Horizon.Common;
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 Result Initialize(ulong address, ulong size)
        {
            Owner = KernelStatic.GetCurrentProcess();

            Result result = Owner.MemoryManager.BorrowCodeMemory(_pageList, address, size);

            if (result != Result.Success)
            {
                return result;
            }

            Owner.CpuMemory.Fill(address, size, 0xff);
            Owner.IncrementReferenceCount();

            _address = address;
            _isMapped = false;
            _isOwnerMapped = false;

            return Result.Success;
        }

        public Result 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();

                Result result = process.MemoryManager.MapPages(address, _pageList, MemoryState.CodeWritable, KMemoryPermission.ReadAndWrite);

                if (result != Result.Success)
                {
                    return result;
                }

                _isMapped = true;
            }

            return Result.Success;
        }

        public Result 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);

                Result result = Owner.MemoryManager.MapPages(address, _pageList, MemoryState.CodeReadOnly, permission);

                if (result != Result.Success)
                {
                    return result;
                }

                _isOwnerMapped = true;
            }

            return Result.Success;
        }

        public Result Unmap(ulong address, ulong size)
        {
            if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, (ulong)KPageTableBase.PageSize))
            {
                return KernelResult.InvalidSize;
            }

            lock (_lock)
            {
                KProcess process = KernelStatic.GetCurrentProcess();

                Result result = process.MemoryManager.UnmapPages(address, _pageList, MemoryState.CodeWritable);

                if (result != Result.Success)
                {
                    return result;
                }

                Debug.Assert(_isMapped);

                _isMapped = false;
            }

            return Result.Success;
        }

        public Result UnmapFromOwner(ulong address, ulong size)
        {
            if (_pageList.GetPagesCount() != BitUtils.DivRoundUp<ulong>(size, KPageTableBase.PageSize))
            {
                return KernelResult.InvalidSize;
            }

            lock (_lock)
            {
                Result result = Owner.MemoryManager.UnmapPages(address, _pageList, MemoryState.CodeReadOnly);

                if (result != Result.Success)
                {
                    return result;
                }

                Debug.Assert(_isOwnerMapped);

                _isOwnerMapped = false;
            }

            return Result.Success;
        }

        protected override void Destroy()
        {
            if (!_isMapped && !_isOwnerMapped)
            {
                ulong size = _pageList.GetPagesCount() * KPageTableBase.PageSize;

                if (Owner.MemoryManager.UnborrowCodeMemory(_address, size, _pageList) != Result.Success)
                {
                    throw new InvalidOperationException("Unexpected failure restoring transfer memory attributes.");
                }
            }

            Owner.DecrementReferenceCount();
        }
    }
}