using System.IO;
namespace Ryujinx.Graphics.Gpu.Shader
{
    /// 
    /// Shader dumper, writes binary shader code to disk.
    /// 
    class ShaderDumper
    {
        private string _runtimeDir;
        private string _dumpPath;
        /// 
        /// Current index of the shader dump binary file.
        /// This is incremented after each save, in order to give unique names to the files.
        /// 
        public int CurrentDumpIndex { get; private set; }
        /// 
        /// Creates a new instance of the shader dumper.
        /// 
        public ShaderDumper()
        {
            CurrentDumpIndex = 1;
        }
        /// 
        /// Dumps shader code to disk.
        /// 
        /// Code to be dumped
        /// True for compute shader code, false for graphics shader code
        /// Paths where the shader code was dumped
        public ShaderDumpPaths Dump(byte[] code, bool compute)
        {
            _dumpPath = GraphicsConfig.ShadersDumpPath;
            if (string.IsNullOrWhiteSpace(_dumpPath))
            {
                return default;
            }
            string fileName = "Shader" + CurrentDumpIndex.ToString("d4") + ".bin";
            string fullPath = Path.Combine(FullDir(), fileName);
            string codePath = Path.Combine(CodeDir(), fileName);
            CurrentDumpIndex++;
            using MemoryStream stream = new MemoryStream(code);
            BinaryReader codeReader = new BinaryReader(stream);
            using FileStream fullFile = File.Create(fullPath);
            using FileStream codeFile = File.Create(codePath);
            BinaryWriter fullWriter = new BinaryWriter(fullFile);
            BinaryWriter codeWriter = new BinaryWriter(codeFile);
            int headerSize = compute ? 0 : 0x50;
            fullWriter.Write(codeReader.ReadBytes(headerSize));
            byte[] temp = codeReader.ReadBytes(code.Length - headerSize);
            fullWriter.Write(temp);
            codeWriter.Write(temp);
            // Align to meet nvdisasm requirements.
            while (codeFile.Length % 0x20 != 0)
            {
                codeWriter.Write(0);
            }
            return new ShaderDumpPaths(fullPath, codePath);
        }
        /// 
        /// Returns the output directory for shader code with header.
        /// 
        /// Directory path
        private string FullDir()
        {
            return CreateAndReturn(Path.Combine(DumpDir(), "Full"));
        }
        /// 
        /// Returns the output directory for shader code without header.
        /// 
        /// Directory path
        private string CodeDir()
        {
            return CreateAndReturn(Path.Combine(DumpDir(), "Code"));
        }
        /// 
        /// Returns the full output directory for the current shader dump.
        /// 
        /// Directory path
        private string DumpDir()
        {
            if (string.IsNullOrEmpty(_runtimeDir))
            {
                int index = 1;
                do
                {
                    _runtimeDir = Path.Combine(_dumpPath, "Dumps" + index.ToString("d2"));
                    index++;
                }
                while (Directory.Exists(_runtimeDir));
                Directory.CreateDirectory(_runtimeDir);
            }
            return _runtimeDir;
        }
        /// 
        /// Creates a new specified directory if needed.
        /// 
        /// The directory to create
        /// The same directory passed to the method
        private static string CreateAndReturn(string dir)
        {
            Directory.CreateDirectory(dir);
            return dir;
        }
    }
}