using Ryujinx.Cpu; using Ryujinx.Cpu.Tracking; using Ryujinx.Memory; using Ryujinx.Memory.Range; using System; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; namespace Ryujinx.Graphics.Gpu.Memory { /// /// Represents physical memory, accessible from the GPU. /// This is actually working CPU virtual addresses, of memory mapped on the application process. /// class PhysicalMemory : IDisposable { public const int PageSize = 0x1000; private IVirtualMemoryManagerTracked _cpuMemory; /// /// Creates a new instance of the physical memory. /// /// CPU memory manager of the application process public PhysicalMemory(IVirtualMemoryManagerTracked cpuMemory) { _cpuMemory = cpuMemory; if (_cpuMemory is IRefCounted rc) { rc.IncrementReferenceCount(); } } /// /// Gets a span of data from the application process. /// /// Start address of the range /// Size in bytes to be range /// True if read tracking is triggered on the span /// A read only span of the data at the specified memory location public ReadOnlySpan GetSpan(ulong address, int size, bool tracked = false) { return _cpuMemory.GetSpan(address, size, tracked); } /// /// Gets a span of data from the application process. /// /// Ranges of physical memory where the data is located /// True if read tracking is triggered on the span /// A read only span of the data at the specified memory location public ReadOnlySpan GetSpan(MultiRange range, bool tracked = false) { if (range.Count == 1) { var singleRange = range.GetSubRange(0); return _cpuMemory.GetSpan(singleRange.Address, (int)singleRange.Size, tracked); } else { Span data = new byte[range.GetSize()]; int offset = 0; for (int i = 0; i < range.Count; i++) { var currentRange = range.GetSubRange(i); int size = (int)currentRange.Size; _cpuMemory.GetSpan(currentRange.Address, size, tracked).CopyTo(data.Slice(offset, size)); offset += size; } return data; } } /// /// Gets a writable region from the application process. /// /// Start address of the range /// Size in bytes to be range /// A writable region with the data at the specified memory location public WritableRegion GetWritableRegion(ulong address, int size) { return _cpuMemory.GetWritableRegion(address, size); } /// /// Reads data from the application process. /// /// Type of the structure /// Address to read from /// The data at the specified memory location public T Read(ulong address) where T : unmanaged { return MemoryMarshal.Cast(GetSpan(address, Unsafe.SizeOf()))[0]; } /// /// Writes data to the application process. /// /// Address to write into /// Data to be written public void Write(ulong address, ReadOnlySpan data) { _cpuMemory.Write(address, data); } /// /// Writes data to the application process. /// /// Ranges of physical memory where the data is located /// Data to be written public void Write(MultiRange range, ReadOnlySpan data) { WriteImpl(range, data, _cpuMemory.Write); } /// /// Writes data to the application process, without any tracking. /// /// Address to write into /// Data to be written public void WriteUntracked(ulong address, ReadOnlySpan data) { _cpuMemory.WriteUntracked(address, data); } /// /// Writes data to the application process, without any tracking. /// /// Ranges of physical memory where the data is located /// Data to be written public void WriteUntracked(MultiRange range, ReadOnlySpan data) { WriteImpl(range, data, _cpuMemory.WriteUntracked); } private delegate void WriteCallback(ulong address, ReadOnlySpan data); /// /// Writes data to the application process, using the supplied callback method. /// /// Ranges of physical memory where the data is located /// Data to be written /// Callback method that will perform the write private void WriteImpl(MultiRange range, ReadOnlySpan data, WriteCallback writeCallback) { if (range.Count == 1) { var singleRange = range.GetSubRange(0); writeCallback(singleRange.Address, data); } else { int offset = 0; for (int i = 0; i < range.Count; i++) { var currentRange = range.GetSubRange(i); int size = (int)currentRange.Size; writeCallback(currentRange.Address, data.Slice(offset, size)); offset += size; } } } /// /// Obtains a memory tracking handle for the given virtual region. This should be disposed when finished with. /// /// CPU virtual address of the region /// Size of the region /// The memory tracking handle public CpuRegionHandle BeginTracking(ulong address, ulong size) { return _cpuMemory.BeginTracking(address, size); } /// /// Obtains a memory tracking handle for the given virtual region. This should be disposed when finished with. /// /// Ranges of physical memory where the data is located /// The memory tracking handle public GpuRegionHandle BeginTracking(MultiRange range) { var cpuRegionHandles = new CpuRegionHandle[range.Count]; for (int i = 0; i < range.Count; i++) { var currentRange = range.GetSubRange(i); cpuRegionHandles[i] = _cpuMemory.BeginTracking(currentRange.Address, currentRange.Size); } return new GpuRegionHandle(cpuRegionHandles); } /// /// Obtains a memory tracking handle for the given virtual region, with a specified granularity. This should be disposed when finished with. /// /// CPU virtual address of the region /// Size of the region /// Desired granularity of write tracking /// The memory tracking handle public CpuMultiRegionHandle BeginGranularTracking(ulong address, ulong size, ulong granularity = 4096) { return _cpuMemory.BeginGranularTracking(address, size, granularity); } /// /// Obtains a smart memory tracking handle for the given virtual region, with a specified granularity. This should be disposed when finished with. /// /// CPU virtual address of the region /// Size of the region /// Desired granularity of write tracking /// The memory tracking handle public CpuSmartMultiRegionHandle BeginSmartGranularTracking(ulong address, ulong size, ulong granularity = 4096) { return _cpuMemory.BeginSmartGranularTracking(address, size, granularity); } /// /// Release our reference to the CPU memory manager. /// public void Dispose() { if (_cpuMemory is IRefCounted rc) { rc.DecrementReferenceCount(); _cpuMemory = null; } } } }