2021-08-11 22:27:00 +02:00
|
|
|
using Ryujinx.Graphics.Shader;
|
2019-10-13 08:02:07 +02:00
|
|
|
using System.IO;
|
|
|
|
|
2019-11-14 19:26:40 +01:00
|
|
|
namespace Ryujinx.Graphics.Gpu.Shader
|
2019-10-13 08:02:07 +02:00
|
|
|
{
|
2019-12-31 05:46:57 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Shader dumper, writes binary shader code to disk.
|
|
|
|
/// </summary>
|
2019-10-13 08:02:07 +02:00
|
|
|
class ShaderDumper
|
|
|
|
{
|
|
|
|
private string _runtimeDir;
|
|
|
|
private string _dumpPath;
|
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Current index of the shader dump binary file.
|
|
|
|
/// This is incremented after each save, in order to give unique names to the files.
|
|
|
|
/// </summary>
|
|
|
|
public int CurrentDumpIndex { get; private set; }
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Creates a new instance of the shader dumper.
|
|
|
|
/// </summary>
|
2019-12-31 05:46:57 +01:00
|
|
|
public ShaderDumper()
|
2019-10-13 08:02:07 +02:00
|
|
|
{
|
2020-05-06 03:02:28 +02:00
|
|
|
CurrentDumpIndex = 1;
|
2019-10-13 08:02:07 +02:00
|
|
|
}
|
|
|
|
|
2019-12-31 05:46:57 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Dumps shader code to disk.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="code">Code to be dumped</param>
|
|
|
|
/// <param name="compute">True for compute shader code, false for graphics shader code</param>
|
2021-08-11 22:27:00 +02:00
|
|
|
/// <returns>Paths where the shader code was dumped</returns>
|
|
|
|
public ShaderDumpPaths Dump(byte[] code, bool compute)
|
2019-10-13 08:02:07 +02:00
|
|
|
{
|
|
|
|
_dumpPath = GraphicsConfig.ShadersDumpPath;
|
|
|
|
|
|
|
|
if (string.IsNullOrWhiteSpace(_dumpPath))
|
|
|
|
{
|
2021-08-11 22:27:00 +02:00
|
|
|
return default;
|
2019-10-13 08:02:07 +02:00
|
|
|
}
|
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
string fileName = "Shader" + CurrentDumpIndex.ToString("d4") + ".bin";
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2021-08-11 22:27:00 +02:00
|
|
|
string fullPath = Path.Combine(FullDir(), fileName);
|
|
|
|
string codePath = Path.Combine(CodeDir(), fileName);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
CurrentDumpIndex++;
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
using MemoryStream stream = new MemoryStream(code);
|
|
|
|
BinaryReader codeReader = new BinaryReader(stream);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
using FileStream fullFile = File.Create(fullPath);
|
|
|
|
using FileStream codeFile = File.Create(codePath);
|
|
|
|
BinaryWriter fullWriter = new BinaryWriter(fullFile);
|
|
|
|
BinaryWriter codeWriter = new BinaryWriter(codeFile);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
int headerSize = compute ? 0 : 0x50;
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
fullWriter.Write(codeReader.ReadBytes(headerSize));
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
byte[] temp = codeReader.ReadBytes(code.Length - headerSize);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
fullWriter.Write(temp);
|
|
|
|
codeWriter.Write(temp);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
// Align to meet nvdisasm requirements.
|
|
|
|
while (codeFile.Length % 0x20 != 0)
|
|
|
|
{
|
|
|
|
codeWriter.Write(0);
|
2019-10-13 08:02:07 +02:00
|
|
|
}
|
2021-08-11 22:27:00 +02:00
|
|
|
|
|
|
|
return new ShaderDumpPaths(fullPath, codePath);
|
2019-10-13 08:02:07 +02:00
|
|
|
}
|
|
|
|
|
2019-12-31 05:46:57 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Returns the output directory for shader code with header.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>Directory path</returns>
|
2019-10-13 08:02:07 +02:00
|
|
|
private string FullDir()
|
|
|
|
{
|
|
|
|
return CreateAndReturn(Path.Combine(DumpDir(), "Full"));
|
|
|
|
}
|
|
|
|
|
2019-12-31 05:46:57 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Returns the output directory for shader code without header.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>Directory path</returns>
|
2019-10-13 08:02:07 +02:00
|
|
|
private string CodeDir()
|
|
|
|
{
|
|
|
|
return CreateAndReturn(Path.Combine(DumpDir(), "Code"));
|
|
|
|
}
|
|
|
|
|
2019-12-31 05:46:57 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Returns the full output directory for the current shader dump.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>Directory path</returns>
|
2019-10-13 08:02:07 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-12-31 05:46:57 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Creates a new specified directory if needed.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="dir">The directory to create</param>
|
|
|
|
/// <returns>The same directory passed to the method</returns>
|
2019-10-13 08:02:07 +02:00
|
|
|
private static string CreateAndReturn(string dir)
|
|
|
|
{
|
|
|
|
Directory.CreateDirectory(dir);
|
|
|
|
|
|
|
|
return dir;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|