310 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			310 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.Common.Collections;
 | |
| using Ryujinx.Common.Logging;
 | |
| using Ryujinx.Graphics.Gpu.Memory;
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| 
 | |
| namespace Ryujinx.HLE.HOS.Services.Nv
 | |
| {
 | |
|     class NvMemoryAllocator
 | |
|     {
 | |
|         private const ulong AddressSpaceSize = 1UL << 40;
 | |
| 
 | |
|         private const ulong DefaultStart = 1UL << 32;
 | |
|         private const ulong InvalidAddress = 0;
 | |
| 
 | |
|         private const ulong PageSize = MemoryManager.PageSize;
 | |
|         private const ulong PageMask = MemoryManager.PageMask;
 | |
| 
 | |
|         public const ulong PteUnmapped = MemoryManager.PteUnmapped;
 | |
| 
 | |
|         // Key   --> Start Address of Region
 | |
|         // Value --> End Address of Region
 | |
|         private readonly TreeDictionary<ulong, ulong> _tree = new TreeDictionary<ulong, ulong>();
 | |
| 
 | |
|         private readonly Dictionary<ulong, LinkedListNode<ulong>> _dictionary = new Dictionary<ulong, LinkedListNode<ulong>>();
 | |
|         private readonly LinkedList<ulong> _list = new LinkedList<ulong>();
 | |
| 
 | |
|         public NvMemoryAllocator()
 | |
|         {
 | |
|             _tree.Add(PageSize, AddressSpaceSize);
 | |
|             LinkedListNode<ulong> node = _list.AddFirst(PageSize);
 | |
|             _dictionary[PageSize] = node;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Marks a range of memory as consumed by removing it from the tree.
 | |
|         /// This function will split memory regions if there is available space.
 | |
|         /// </summary>
 | |
|         /// <param name="va">Virtual address at which to allocate</param>
 | |
|         /// <param name="size">Size of the allocation in bytes</param>
 | |
|         /// <param name="referenceAddress">Reference to the address of memory where the allocation can take place</param>
 | |
|         #region Memory Allocation
 | |
|         public void AllocateRange(ulong va, ulong size, ulong referenceAddress = InvalidAddress)
 | |
|         {
 | |
|             lock (_tree)
 | |
|             {
 | |
|                 Logger.Debug?.Print(LogClass.ServiceNv, $"Allocating range from 0x{va:X} to 0x{(va + size):X}.");
 | |
|                 if (referenceAddress != InvalidAddress)
 | |
|                 {
 | |
|                     ulong endAddress = va + size;
 | |
|                     ulong referenceEndAddress = _tree.Get(referenceAddress);
 | |
|                     if (va >= referenceAddress)
 | |
|                     {
 | |
|                         // Need Left Node
 | |
|                         if (va > referenceAddress)
 | |
|                         {
 | |
|                             ulong leftEndAddress = va;
 | |
| 
 | |
|                             // Overwrite existing block with its new smaller range.
 | |
|                             _tree.Add(referenceAddress, leftEndAddress);
 | |
|                             Logger.Debug?.Print(LogClass.ServiceNv, $"Created smaller address range from 0x{referenceAddress:X} to 0x{leftEndAddress:X}.");
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             // We need to get rid of the large chunk.
 | |
|                             _tree.Remove(referenceAddress);
 | |
|                         }
 | |
| 
 | |
|                         ulong rightSize = referenceEndAddress - endAddress;
 | |
|                         // If leftover space, create a right node.
 | |
|                         if (rightSize > 0)
 | |
|                         {
 | |
|                             Logger.Debug?.Print(LogClass.ServiceNv, $"Created smaller address range from 0x{endAddress:X} to 0x{referenceEndAddress:X}.");
 | |
|                             _tree.Add(endAddress, referenceEndAddress);
 | |
| 
 | |
|                             LinkedListNode<ulong> node = _list.AddAfter(_dictionary[referenceAddress], endAddress);
 | |
|                             _dictionary[endAddress] = node;
 | |
|                         }
 | |
| 
 | |
|                         if (va == referenceAddress)
 | |
|                         {
 | |
|                             _list.Remove(_dictionary[referenceAddress]);
 | |
|                             _dictionary.Remove(referenceAddress);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Marks a range of memory as free by adding it to the tree.
 | |
|         /// This function will automatically compact the tree when it determines there are multiple ranges of free memory adjacent to each other.
 | |
|         /// </summary>
 | |
|         /// <param name="va">Virtual address at which to deallocate</param>
 | |
|         /// <param name="size">Size of the allocation in bytes</param>
 | |
|         public void DeallocateRange(ulong va, ulong size)
 | |
|         {
 | |
|             lock (_tree)
 | |
|             {
 | |
|                 Logger.Debug?.Print(LogClass.ServiceNv, $"Deallocating address range from 0x{va:X} to 0x{(va + size):X}.");
 | |
| 
 | |
|                 ulong freeAddressStartPosition = _tree.Floor(va);
 | |
|                 if (freeAddressStartPosition != InvalidAddress)
 | |
|                 {
 | |
|                     LinkedListNode<ulong> node = _dictionary[freeAddressStartPosition];
 | |
|                     ulong targetPrevAddress = _dictionary[freeAddressStartPosition].Previous != null ? _dictionary[_dictionary[freeAddressStartPosition].Previous.Value].Value : InvalidAddress;
 | |
|                     ulong targetNextAddress = _dictionary[freeAddressStartPosition].Next != null ? _dictionary[_dictionary[freeAddressStartPosition].Next.Value].Value : InvalidAddress;
 | |
|                     ulong expandedStart = va;
 | |
|                     ulong expandedEnd = va + size;
 | |
| 
 | |
|                     while (targetPrevAddress != InvalidAddress)
 | |
|                     {
 | |
|                         ulong prevAddress = targetPrevAddress;
 | |
|                         ulong prevEndAddress = _tree.Get(targetPrevAddress);
 | |
|                         if (prevEndAddress >= expandedStart)
 | |
|                         {
 | |
|                             expandedStart = targetPrevAddress;
 | |
|                             LinkedListNode<ulong> prevPtr = _dictionary[prevAddress];
 | |
|                             if (prevPtr.Previous != null)
 | |
|                             {
 | |
|                                 targetPrevAddress = prevPtr.Previous.Value;
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 targetPrevAddress = InvalidAddress;
 | |
|                             }
 | |
|                             node = node.Previous;
 | |
|                             _tree.Remove(prevAddress);
 | |
|                             _list.Remove(_dictionary[prevAddress]);
 | |
|                             _dictionary.Remove(prevAddress);
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             break;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     while (targetNextAddress != InvalidAddress)
 | |
|                     {
 | |
|                         ulong nextAddress = targetNextAddress;
 | |
|                         ulong nextEndAddress = _tree.Get(targetNextAddress);
 | |
|                         if (nextAddress <= expandedEnd)
 | |
|                         {
 | |
|                             expandedEnd = Math.Max(expandedEnd, nextEndAddress);
 | |
|                             LinkedListNode<ulong> nextPtr = _dictionary[nextAddress];
 | |
|                             if (nextPtr.Next != null)
 | |
|                             {
 | |
|                                 targetNextAddress = nextPtr.Next.Value;
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 targetNextAddress = InvalidAddress;
 | |
|                             }
 | |
|                             _tree.Remove(nextAddress);
 | |
|                             _list.Remove(_dictionary[nextAddress]);
 | |
|                             _dictionary.Remove(nextAddress);
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             break;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     Logger.Debug?.Print(LogClass.ServiceNv, $"Deallocation resulted in new free range from 0x{expandedStart:X} to 0x{expandedEnd:X}.");
 | |
| 
 | |
|                     _tree.Add(expandedStart, expandedEnd);
 | |
|                     LinkedListNode<ulong> nodePtr = _list.AddAfter(node, expandedStart);
 | |
|                     _dictionary[expandedStart] = nodePtr;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Gets the address of an unused (free) region of the specified size.
 | |
|         /// </summary>
 | |
|         /// <param name="size">Size of the region in bytes</param>
 | |
|         /// <param name="freeAddressStartPosition">Position at which memory can be allocated</param>
 | |
|         /// <param name="alignment">Required alignment of the region address in bytes</param>
 | |
|         /// <param name="start">Start address of the search on the address space</param>
 | |
|         /// <returns>GPU virtual address of the allocation, or an all ones mask in case of failure</returns>
 | |
|         public ulong GetFreeAddress(ulong size, out ulong freeAddressStartPosition, ulong alignment = 1, ulong start = DefaultStart)
 | |
|         {
 | |
|             // Note: Address 0 is not considered valid by the driver,
 | |
|             // when 0 is returned it's considered a mapping error.
 | |
|             lock (_tree)
 | |
|             {
 | |
|                 Logger.Debug?.Print(LogClass.ServiceNv, $"Searching for a free address @ 0x{start:X} of size 0x{size:X}.");
 | |
|                 ulong address = start;
 | |
| 
 | |
|                 if (alignment == 0)
 | |
|                 {
 | |
|                     alignment = 1;
 | |
|                 }
 | |
| 
 | |
|                 alignment = (alignment + PageMask) & ~PageMask;
 | |
|                 if (address < AddressSpaceSize)
 | |
|                 {
 | |
|                     bool reachedEndOfAddresses = false;
 | |
|                     ulong targetAddress;
 | |
|                     if (start == DefaultStart)
 | |
|                     {
 | |
|                         Logger.Debug?.Print(LogClass.ServiceNv, $"Target address set to start of the last available range: 0x{_list.Last.Value:X}.");
 | |
|                         targetAddress = _list.Last.Value;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         targetAddress = _tree.Floor(address);
 | |
|                         Logger.Debug?.Print(LogClass.ServiceNv, $"Target address set to floor of 0x{address:X}; resulted in 0x{targetAddress:X}.");
 | |
|                         if (targetAddress == InvalidAddress)
 | |
|                         {
 | |
|                             targetAddress = _tree.Ceiling(address);
 | |
|                             Logger.Debug?.Print(LogClass.ServiceNv, $"Target address was invalid, set to ceiling of 0x{address:X}; resulted in 0x{targetAddress:X}");
 | |
|                         }
 | |
|                     }
 | |
|                     while (address < AddressSpaceSize)
 | |
|                     {
 | |
|                         if (targetAddress != InvalidAddress)
 | |
|                         {
 | |
|                             if (address >= targetAddress)
 | |
|                             {
 | |
|                                 if (address + size <= _tree.Get(targetAddress))
 | |
|                                 {
 | |
|                                     Logger.Debug?.Print(LogClass.ServiceNv, $"Found a suitable free address range from 0x{targetAddress:X} to 0x{_tree.Get(targetAddress):X} for 0x{address:X}.");
 | |
|                                     freeAddressStartPosition = targetAddress;
 | |
|                                     return address;
 | |
|                                 }
 | |
|                                 else
 | |
|                                 {
 | |
|                                     Logger.Debug?.Print(LogClass.ServiceNv, "Address requirements exceeded the available space in the target range.");
 | |
|                                     LinkedListNode<ulong> nextPtr = _dictionary[targetAddress];
 | |
|                                     if (nextPtr.Next != null)
 | |
|                                     {
 | |
|                                         targetAddress = nextPtr.Next.Value;
 | |
|                                         Logger.Debug?.Print(LogClass.ServiceNv, $"Moved search to successor range starting at 0x{targetAddress:X}.");
 | |
|                                     }
 | |
|                                     else
 | |
|                                     {
 | |
|                                         if (reachedEndOfAddresses)
 | |
|                                         {
 | |
|                                             Logger.Debug?.Print(LogClass.ServiceNv, "Exiting loop, a full pass has already been completed w/ no suitable free address range.");
 | |
|                                             break;
 | |
|                                         }
 | |
|                                         else
 | |
|                                         {
 | |
|                                             reachedEndOfAddresses = true;
 | |
|                                             address = start;
 | |
|                                             targetAddress = _tree.Floor(address);
 | |
|                                             Logger.Debug?.Print(LogClass.ServiceNv, $"Reached the end of the available free ranges, restarting loop @ 0x{targetAddress:X} for 0x{address:X}.");
 | |
|                                         }
 | |
|                                     }
 | |
|                                 }
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 address += PageSize * (targetAddress / PageSize - (address / PageSize));
 | |
| 
 | |
|                                 ulong remainder = address % alignment;
 | |
| 
 | |
|                                 if (remainder != 0)
 | |
|                                 {
 | |
|                                     address = (address - remainder) + alignment;
 | |
|                                 }
 | |
| 
 | |
|                                 Logger.Debug?.Print(LogClass.ServiceNv, $"Reset and aligned address to {address:X}.");
 | |
| 
 | |
|                                 if (address + size > AddressSpaceSize && !reachedEndOfAddresses)
 | |
|                                 {
 | |
|                                     reachedEndOfAddresses = true;
 | |
|                                     address = start;
 | |
|                                     targetAddress = _tree.Floor(address);
 | |
|                                     Logger.Debug?.Print(LogClass.ServiceNv, $"Address requirements exceeded the capacity of available address space, restarting loop @ 0x{targetAddress:X} for 0x{address:X}.");
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             break;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 Logger.Debug?.Print(LogClass.ServiceNv, $"No suitable address range found; returning: 0x{InvalidAddress:X}.");
 | |
|                 freeAddressStartPosition = InvalidAddress;
 | |
|             }
 | |
| 
 | |
|             return PteUnmapped;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Checks if a given memory region is mapped or reserved.
 | |
|         /// </summary>
 | |
|         /// <param name="gpuVa">GPU virtual address of the page</param>
 | |
|         /// <param name="size">Size of the allocation in bytes</param>
 | |
|         /// <param name="freeAddressStartPosition">Nearest lower address that memory can be allocated</param>
 | |
|         /// <returns>True if the page is mapped or reserved, false otherwise</returns>
 | |
|         public bool IsRegionInUse(ulong gpuVa, ulong size, out ulong freeAddressStartPosition)
 | |
|         {
 | |
|             lock (_tree)
 | |
|             {
 | |
|                 ulong floorAddress = _tree.Floor(gpuVa);
 | |
|                 freeAddressStartPosition = floorAddress;
 | |
|                 if (floorAddress != InvalidAddress)
 | |
|                 {
 | |
|                     return !(gpuVa >= floorAddress && ((gpuVa + size) <= _tree.Get(floorAddress)));
 | |
|                 }
 | |
|             }
 | |
|             return true;
 | |
|         }
 | |
|         #endregion
 | |
|     }
 | |
| }
 | 
