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
    {
        public const int PageSize = 0x1000;
        private readonly Cpu.MemoryManager _cpuMemory;
        /// 
        /// Creates a new instance of the physical memory.
        /// 
        /// CPU memory manager of the application process
        public PhysicalMemory(Cpu.MemoryManager cpuMemory)
        {
            _cpuMemory = cpuMemory;
        }
        /// 
        /// 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);
        }
    }
}