using Ryujinx.Common.Logging;
using Ryujinx.Graphics.GAL;
using Ryujinx.Graphics.Gpu.Memory;
using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
namespace Ryujinx.Graphics.Gpu.Image
{
    /// 
    /// Texture pool.
    /// 
    class TexturePool : Pool
    {
        private int _sequenceNumber;
        /// 
        /// Intrusive linked list node used on the texture pool cache.
        /// 
        public LinkedListNode CacheNode { get; set; }
        /// 
        /// Constructs a new instance of the texture pool.
        /// 
        /// GPU context that the texture pool belongs to
        /// Address of the texture pool in guest memory
        /// Maximum texture ID of the texture pool (equal to maximum textures minus one)
        public TexturePool(GpuContext context, ulong address, int maximumId) : base(context, address, maximumId) { }
        /// 
        /// Gets the texture with the given ID.
        /// 
        /// ID of the texture. This is effectively a zero-based index
        /// The texture with the given ID
        public override Texture Get(int id)
        {
            if ((uint)id >= Items.Length)
            {
                return null;
            }
            if (_sequenceNumber != Context.SequenceNumber)
            {
                _sequenceNumber = Context.SequenceNumber;
                SynchronizeMemory();
            }
            Texture texture = Items[id];
            if (texture == null)
            {
                TextureDescriptor descriptor = GetDescriptor(id);
                TextureInfo info = GetInfo(descriptor);
                // Bad address. We can't add a texture with a invalid address
                // to the cache.
                if (info.Address == MemoryManager.BadAddress)
                {
                    return null;
                }
                texture = Context.Methods.TextureManager.FindOrCreateTexture(info, TextureSearchFlags.Sampler);
                texture.IncrementReferenceCount();
                Items[id] = texture;
            }
            else
            {
                // Memory is automatically synchronized on texture creation.
                texture.SynchronizeMemory();
            }
            return texture;
        }
        /// 
        /// Gets the texture descriptor from a given texture ID.
        /// 
        /// ID of the texture. This is effectively a zero-based index
        /// The texture descriptor
        public TextureDescriptor GetDescriptor(int id)
        {
            ulong address = Address + (ulong)(uint)id * DescriptorSize;
            ReadOnlySpan data = Context.PhysicalMemory.GetSpan(address, DescriptorSize);
            return MemoryMarshal.Cast(data)[0];
        }
        /// 
        /// Implementation of the texture pool range invalidation.
        /// 
        /// Start address of the range of the texture pool
        /// Size of the range being invalidated
        protected override void InvalidateRangeImpl(ulong address, ulong size)
        {
            ulong endAddress = address + size;
            for (; address < endAddress; address += DescriptorSize)
            {
                int id = (int)((address - Address) / DescriptorSize);
                Texture texture = Items[id];
                if (texture != null)
                {
                    ReadOnlySpan data = Context.PhysicalMemory.GetSpan(address, DescriptorSize);
                    TextureDescriptor descriptor = MemoryMarshal.Cast(data)[0];
                    // If the descriptors are the same, the texture is the same,
                    // we don't need to remove as it was not modified. Just continue.
                    if (texture.IsPerfectMatch(GetInfo(descriptor), TextureSearchFlags.Strict))
                    {
                        continue;
                    }
                    texture.DecrementReferenceCount();
                    Items[id] = null;
                }
            }
        }
        /// 
        /// Gets texture information from a texture descriptor.
        /// 
        /// The texture descriptor
        /// The texture information
        private TextureInfo GetInfo(TextureDescriptor descriptor)
        {
            ulong address = Context.MemoryManager.Translate(descriptor.UnpackAddress());
            int width         = descriptor.UnpackWidth();
            int height        = descriptor.UnpackHeight();
            int depthOrLayers = descriptor.UnpackDepth();
            int levels        = descriptor.UnpackLevels();
            TextureMsaaMode msaaMode = descriptor.UnpackTextureMsaaMode();
            int samplesInX = msaaMode.SamplesInX();
            int samplesInY = msaaMode.SamplesInY();
            int stride = descriptor.UnpackStride();
            TextureDescriptorType descriptorType = descriptor.UnpackTextureDescriptorType();
            bool isLinear = descriptorType == TextureDescriptorType.Linear;
            Target target = descriptor.UnpackTextureTarget().Convert((samplesInX | samplesInY) != 1);
            uint format = descriptor.UnpackFormat();
            bool srgb   = descriptor.UnpackSrgb();
            if (!FormatTable.TryGetTextureFormat(format, srgb, out FormatInfo formatInfo))
            {
                Logger.PrintDebug(LogClass.Gpu, $"Invalid texture format 0x{format:X} (sRGB: {srgb}).");
                formatInfo = FormatInfo.Default;
            }
            int gobBlocksInY = descriptor.UnpackGobBlocksInY();
            int gobBlocksInZ = descriptor.UnpackGobBlocksInZ();
            int gobBlocksInTileX = descriptor.UnpackGobBlocksInTileX();
            SwizzleComponent swizzleR = descriptor.UnpackSwizzleR().Convert();
            SwizzleComponent swizzleG = descriptor.UnpackSwizzleG().Convert();
            SwizzleComponent swizzleB = descriptor.UnpackSwizzleB().Convert();
            SwizzleComponent swizzleA = descriptor.UnpackSwizzleA().Convert();
            DepthStencilMode depthStencilMode = GetDepthStencilMode(
                formatInfo.Format,
                swizzleR,
                swizzleG,
                swizzleB,
                swizzleA);
            return new TextureInfo(
                address,
                width,
                height,
                depthOrLayers,
                levels,
                samplesInX,
                samplesInY,
                stride,
                isLinear,
                gobBlocksInY,
                gobBlocksInZ,
                gobBlocksInTileX,
                target,
                formatInfo,
                depthStencilMode,
                swizzleR,
                swizzleG,
                swizzleB,
                swizzleA);
        }
        /// 
        /// Gets the texture depth-stencil mode, based on the swizzle components of each color channel.
        /// The depth-stencil mode is determined based on how the driver sets those parameters.
        /// 
        /// The format of the texture
        /// The texture swizzle components
        /// The depth-stencil mode
        private static DepthStencilMode GetDepthStencilMode(Format format, params SwizzleComponent[] components)
        {
            // R = Depth, G = Stencil.
            // On 24-bits depth formats, this is inverted (Stencil is R etc).
            // NVN setup:
            // For depth, A is set to 1.0f, the other components are set to Depth.
            // For stencil, all components are set to Stencil.
            SwizzleComponent component = components[0];
            for (int index = 1; index < 4 && !IsRG(component); index++)
            {
                component = components[index];
            }
            if (!IsRG(component))
            {
                return DepthStencilMode.Depth;
            }
            if (format == Format.D24X8Unorm || format == Format.D24UnormS8Uint)
            {
                return component == SwizzleComponent.Red
                    ? DepthStencilMode.Stencil
                    : DepthStencilMode.Depth;
            }
            else
            {
                return component == SwizzleComponent.Red
                    ? DepthStencilMode.Depth
                    : DepthStencilMode.Stencil;
            }
        }
        /// 
        /// Checks if the swizzle component is equal to the red or green channels.
        /// 
        /// The swizzle component to check
        /// True if the swizzle component is equal to the red or green, false otherwise
        private static bool IsRG(SwizzleComponent component)
        {
            return component == SwizzleComponent.Red ||
                   component == SwizzleComponent.Green;
        }
        /// 
        /// Decrements the reference count of the texture.
        /// This indicates that the texture pool is not using it anymore.
        /// 
        /// The texture to be deleted
        protected override void Delete(Texture item)
        {
            item?.DecrementReferenceCount();
        }
    }
}