2012-11-15 00:39:56 +01:00
|
|
|
#include "stdafx.h"
|
|
|
|
|
#include "FragmentProgram.h"
|
|
|
|
|
|
2012-11-15 16:45:11 +01:00
|
|
|
void FragmentDecompilerThread::AddCode(wxString code)
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
|
|
|
|
if(!src0.exec_if_eq && !src0.exec_if_gr && !src0.exec_if_le) return;
|
|
|
|
|
if(!src0.exec_if_eq || !src0.exec_if_gr || !src0.exec_if_le)
|
|
|
|
|
{
|
|
|
|
|
ConLog.Error("Bad cond! eq: %d gr: %d le: %d", src0.exec_if_eq, src0.exec_if_gr, src0.exec_if_le);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(src1.scale)
|
|
|
|
|
{
|
|
|
|
|
switch(src1.scale)
|
|
|
|
|
{
|
|
|
|
|
case 1: code = "(" + code + ") * 2"; break;
|
|
|
|
|
case 2: code = "(" + code + ") * 4"; break;
|
|
|
|
|
case 3: code = "(" + code + ") * 8"; break;
|
|
|
|
|
case 5: code = "(" + code + ") / 2"; break;
|
|
|
|
|
case 6: code = "(" + code + ") / 4"; break;
|
|
|
|
|
case 7: code = "(" + code + ") / 8"; break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
ConLog.Error("Bad scale: %d", src1.scale);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-15 16:45:11 +01:00
|
|
|
code = AddReg(dst.dest_reg) + GetMask() + " = " + code + GetMask();
|
2012-11-15 00:39:56 +01:00
|
|
|
|
|
|
|
|
main += "\t" + code + ";\n";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wxString FragmentDecompilerThread::GetMask()
|
|
|
|
|
{
|
|
|
|
|
wxString ret = wxEmptyString;
|
|
|
|
|
|
|
|
|
|
if(dst.mask_x) ret += 'x';
|
|
|
|
|
if(dst.mask_y) ret += 'y';
|
|
|
|
|
if(dst.mask_z) ret += 'z';
|
|
|
|
|
if(dst.mask_w) ret += 'w';
|
|
|
|
|
|
|
|
|
|
return ret.IsEmpty() || ret == "xyzw" ? wxEmptyString : ("." + ret);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wxString FragmentDecompilerThread::AddReg(u32 index)
|
|
|
|
|
{
|
|
|
|
|
//if(!index) return "gl_FragColor";
|
2013-06-30 10:46:29 +02:00
|
|
|
return m_parr.AddParam(index ? PARAM_NONE : PARAM_OUT, "vec4", wxString::Format("r%d", index), index ? -1 : 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wxString FragmentDecompilerThread::AddConst()
|
|
|
|
|
{
|
|
|
|
|
return m_parr.AddParam(PARAM_CONST, "vec4", wxString::Format("fc%d", m_const_index++));
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wxString FragmentDecompilerThread::AddTex()
|
|
|
|
|
{
|
|
|
|
|
return m_parr.AddParam(PARAM_CONST, "sampler2D", wxString::Format("tex_%d", dst.tex_num));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<typename T> wxString FragmentDecompilerThread::GetSRC(T src)
|
|
|
|
|
{
|
|
|
|
|
wxString ret = wxEmptyString;
|
|
|
|
|
|
|
|
|
|
switch(src.reg_type)
|
|
|
|
|
{
|
|
|
|
|
case 0: //tmp
|
|
|
|
|
ret += AddReg(src.tmp_reg_index);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 1: //input
|
|
|
|
|
{
|
|
|
|
|
static const wxString reg_table[] =
|
|
|
|
|
{
|
|
|
|
|
"gl_Position",
|
|
|
|
|
"col0", "col1",
|
|
|
|
|
"fogc",
|
|
|
|
|
"tc0", "tc1", "tc2", "tc3", "tc4", "tc5", "tc6", "tc7"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
switch(dst.src_attr_reg_num)
|
|
|
|
|
{
|
|
|
|
|
case 0x00: ret += reg_table[0]; break;
|
|
|
|
|
default:
|
|
|
|
|
if(dst.src_attr_reg_num < WXSIZEOF(reg_table))
|
|
|
|
|
{
|
|
|
|
|
ret += m_parr.AddParam(PARAM_IN, "vec4", reg_table[dst.src_attr_reg_num]);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
ConLog.Error("Bad src reg num: %d", dst.src_attr_reg_num);
|
|
|
|
|
ret += m_parr.AddParam(PARAM_IN, "vec4", "unk");
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
2013-06-30 10:46:29 +02:00
|
|
|
case 2: //const
|
|
|
|
|
ConLog.Write("reg index = %d", src.tmp_reg_index);
|
|
|
|
|
ret += AddConst();
|
|
|
|
|
break;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
ConLog.Error("Bad src type %d", src.reg_type);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const char f[4] = {'x', 'y', 'z', 'w'};
|
|
|
|
|
wxString swizzle = wxEmptyString;
|
|
|
|
|
swizzle += f[src.swizzle_x];
|
|
|
|
|
swizzle += f[src.swizzle_y];
|
|
|
|
|
swizzle += f[src.swizzle_z];
|
|
|
|
|
swizzle += f[src.swizzle_w];
|
|
|
|
|
|
|
|
|
|
if(swizzle != "xyzw") ret += "." + swizzle;
|
|
|
|
|
|
|
|
|
|
if(src.abs) ret = "abs(" + ret + ")";
|
|
|
|
|
if(src.neg) ret = "-" + ret;
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wxString FragmentDecompilerThread::BuildCode()
|
|
|
|
|
{
|
|
|
|
|
wxString p = wxEmptyString;
|
|
|
|
|
|
|
|
|
|
for(u32 i=0; i<m_parr.params.GetCount(); ++i)
|
|
|
|
|
{
|
2013-06-30 10:46:29 +02:00
|
|
|
p += m_parr.params[i].Format();
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const wxString& prot =
|
2013-06-30 10:46:29 +02:00
|
|
|
"#version 330\n"
|
2012-11-15 00:39:56 +01:00
|
|
|
"\n"
|
|
|
|
|
"%s\n"
|
|
|
|
|
"void main()\n{\n%s}\n";
|
|
|
|
|
|
|
|
|
|
return wxString::Format(prot, p, main);
|
|
|
|
|
}
|
|
|
|
|
|
2013-06-30 10:46:29 +02:00
|
|
|
void FragmentDecompilerThread::Task()
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
|
|
|
|
mem32_t data(m_addr);
|
2013-06-30 10:46:29 +02:00
|
|
|
m_size = 0;
|
2012-11-15 00:39:56 +01:00
|
|
|
|
2013-06-30 10:46:29 +02:00
|
|
|
while(true)
|
2012-11-15 00:39:56 +01:00
|
|
|
{
|
|
|
|
|
dst.HEX = GetData(data[0]);
|
|
|
|
|
src0.HEX = GetData(data[1]);
|
|
|
|
|
src1.HEX = GetData(data[2]);
|
|
|
|
|
src2.HEX = GetData(data[3]);
|
|
|
|
|
|
|
|
|
|
switch(dst.opcode)
|
|
|
|
|
{
|
2012-11-15 16:45:11 +01:00
|
|
|
case 0x00: break; //NOP
|
|
|
|
|
case 0x01: AddCode(GetSRC(src0)); break; //MOV
|
|
|
|
|
case 0x02: AddCode("(" + GetSRC(src0) + " * " + GetSRC(src1) + ")"); break; //MUL
|
2013-06-30 10:46:29 +02:00
|
|
|
case 0x03: AddCode("(" + GetSRC(src0) + " + " + GetSRC(src1) + ")"); break; //ADD
|
|
|
|
|
case 0x04: AddCode("(" + GetSRC(src0) + " * " + GetSRC(src1) + " + " + GetSRC(src2) + ")"); break; //MAD
|
|
|
|
|
case 0x05: AddCode("dot(" + GetSRC(src0) + ".xyz, " + GetSRC(src1) + ".xyz)"); break; // DP3
|
|
|
|
|
case 0x06: AddCode("dot(" + GetSRC(src0) + ", " + GetSRC(src1) + ")"); break; // DP4
|
|
|
|
|
//case 0x07: break; // DST
|
|
|
|
|
case 0x08: AddCode("min(" + GetSRC(src0) + ", " + GetSRC(src1) + ")"); break; // MIN
|
|
|
|
|
case 0x09: AddCode("max(" + GetSRC(src0) + ", " + GetSRC(src1) + ")"); break; // MAX
|
|
|
|
|
//case 0x0a: break; // SLT
|
|
|
|
|
//case 0x0b: break; // SGE
|
|
|
|
|
//case 0x0c: break; // SLE
|
|
|
|
|
//case 0x0d: break; // SGT
|
|
|
|
|
//case 0x0e: break; // SNE
|
|
|
|
|
//case 0x0f: break; // SEQ
|
|
|
|
|
|
|
|
|
|
//case 0x10: break; // FRC
|
|
|
|
|
//case 0x11: break; // FLR
|
|
|
|
|
//case 0x12: break; // KIL
|
|
|
|
|
//case 0x13: break; // PK4
|
|
|
|
|
//case 0x14: break; // UP4
|
|
|
|
|
case 0x15: AddCode("ddx(" + GetSRC(src0) + ")"); break; // DDX
|
|
|
|
|
case 0x16: AddCode("ddy(" + GetSRC(src0) + ")"); break; // DDY
|
2012-11-15 16:45:11 +01:00
|
|
|
case 0x17: AddCode("texture(" + AddTex() + ", (" + GetSRC(src0) + ").xy)"); break; //TEX
|
2013-06-30 10:46:29 +02:00
|
|
|
//case 0x18: break; // TXP
|
|
|
|
|
//case 0x19: break; // TXD
|
|
|
|
|
case 0x1a: AddCode("1 / (" + GetSRC(src0) + ")"); break; // RCP
|
|
|
|
|
case 0x1b: AddCode("inversesqrt(" + GetSRC(src0) + ")"); break; // RSQ
|
|
|
|
|
case 0x1c: AddCode("exp2(" + GetSRC(src0) + ")"); break; // EX2
|
|
|
|
|
case 0x1d: AddCode("log2(" + GetSRC(src0) + ")"); break; // LG2
|
|
|
|
|
//case 0x1e: break; // LIT
|
|
|
|
|
//case 0x1f: break; // LRP
|
|
|
|
|
|
|
|
|
|
//case 0x20: break; // STR
|
|
|
|
|
//case 0x21: break; // SFL
|
|
|
|
|
case 0x22: AddCode("cos(" + GetSRC(src0) + ")"); break; // COS
|
|
|
|
|
case 0x23: AddCode("sin(" + GetSRC(src0) + ")"); break; // SIN
|
|
|
|
|
//case 0x24: break; // PK2
|
|
|
|
|
//case 0x25: break; // UP2
|
|
|
|
|
//case 0x26: break; // POW
|
|
|
|
|
//case 0x27: break; // PKB
|
|
|
|
|
//case 0x28: break; // UPB
|
|
|
|
|
//case 0x29: break; // PK16
|
|
|
|
|
//case 0x2a: break; // UP16
|
|
|
|
|
//case 0x2b: break; // BEM
|
|
|
|
|
//case 0x2c: break; // PKG
|
|
|
|
|
//case 0x2d: break; // UPG
|
|
|
|
|
//case 0x2e: break; // DP2A
|
|
|
|
|
//case 0x2f: break; // TXL
|
|
|
|
|
|
|
|
|
|
//case 0x31: break; // TXB
|
|
|
|
|
//case 0x33: break; // TEXBEM
|
|
|
|
|
//case 0x34: break; // TXPBEM
|
|
|
|
|
//case 0x35: break; // BEMLUM
|
|
|
|
|
//case 0x36: break; // REFL
|
|
|
|
|
//case 0x37: break; // TIMESWTEX
|
|
|
|
|
//case 0x38: break; // DP2
|
|
|
|
|
//case 0x39: break; // NRM
|
|
|
|
|
//case 0x3a: break; // DIV
|
|
|
|
|
//case 0x3b: break; // DIVSQ
|
|
|
|
|
//case 0x3c: break; // LIF
|
|
|
|
|
//case 0x3d: break; // FENCT
|
|
|
|
|
//case 0x3e: break; // FENCB
|
2012-11-15 00:39:56 +01:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
ConLog.Error("Unknown opcode 0x%x", dst.opcode);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_size += 4 * 4;
|
|
|
|
|
|
|
|
|
|
if(dst.end) break;
|
|
|
|
|
|
|
|
|
|
data.SetOffset(4 * 4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_shader = BuildCode();
|
|
|
|
|
main.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ShaderProgram::ShaderProgram()
|
|
|
|
|
: m_decompiler_thread(NULL)
|
|
|
|
|
, id(0)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ShaderProgram::~ShaderProgram()
|
|
|
|
|
{
|
|
|
|
|
if(m_decompiler_thread)
|
|
|
|
|
{
|
|
|
|
|
Wait();
|
2013-06-30 10:46:29 +02:00
|
|
|
if(m_decompiler_thread->IsAlive())
|
|
|
|
|
{
|
|
|
|
|
m_decompiler_thread->Stop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
delete m_decompiler_thread;
|
|
|
|
|
m_decompiler_thread = nullptr;
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Delete();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ShaderProgram::Decompile()
|
|
|
|
|
{
|
|
|
|
|
#if 0
|
|
|
|
|
FragmentDecompilerThread(shader, parr, addr).Entry();
|
|
|
|
|
#else
|
|
|
|
|
if(m_decompiler_thread)
|
|
|
|
|
{
|
|
|
|
|
Wait();
|
2013-06-30 10:46:29 +02:00
|
|
|
if(m_decompiler_thread->IsAlive())
|
|
|
|
|
{
|
|
|
|
|
m_decompiler_thread->Stop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
delete m_decompiler_thread;
|
|
|
|
|
m_decompiler_thread = nullptr;
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_decompiler_thread = new FragmentDecompilerThread(shader, parr, addr, size);
|
2013-06-30 10:46:29 +02:00
|
|
|
m_decompiler_thread->Start();
|
2012-11-15 00:39:56 +01:00
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ShaderProgram::Compile()
|
|
|
|
|
{
|
|
|
|
|
if(id) glDeleteShader(id);
|
|
|
|
|
|
|
|
|
|
id = glCreateShader(GL_FRAGMENT_SHADER);
|
|
|
|
|
|
|
|
|
|
const char* str = shader.c_str();
|
|
|
|
|
const int strlen = shader.Len();
|
|
|
|
|
|
|
|
|
|
glShaderSource(id, 1, &str, &strlen);
|
|
|
|
|
glCompileShader(id);
|
|
|
|
|
|
|
|
|
|
GLint r = GL_FALSE;
|
|
|
|
|
glGetShaderiv(id, GL_COMPILE_STATUS, &r);
|
|
|
|
|
if(r != GL_TRUE)
|
|
|
|
|
{
|
|
|
|
|
glGetShaderiv(id, GL_INFO_LOG_LENGTH, &r);
|
|
|
|
|
|
|
|
|
|
if(r)
|
|
|
|
|
{
|
|
|
|
|
char* buf = new char[r+1];
|
|
|
|
|
GLsizei len;
|
|
|
|
|
memset(buf, 0, r+1);
|
|
|
|
|
glGetShaderInfoLog(id, r, &len, buf);
|
|
|
|
|
ConLog.Error("Failed to compile shader: %s", buf);
|
2013-06-30 10:46:29 +02:00
|
|
|
delete[] buf;
|
2012-11-15 00:39:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ConLog.Write(shader);
|
|
|
|
|
Emu.Pause();
|
|
|
|
|
}
|
|
|
|
|
//else ConLog.Write("Shader compiled successfully!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ShaderProgram::Delete()
|
|
|
|
|
{
|
|
|
|
|
for(u32 i=0; i<parr.params.GetCount(); ++i)
|
|
|
|
|
{
|
2013-06-30 10:46:29 +02:00
|
|
|
parr.params[i].items.Clear();
|
2012-11-15 00:39:56 +01:00
|
|
|
parr.params[i].type.Clear();
|
|
|
|
|
}
|
2013-06-30 10:46:29 +02:00
|
|
|
|
2012-11-15 00:39:56 +01:00
|
|
|
parr.params.Clear();
|
|
|
|
|
shader.Clear();
|
|
|
|
|
|
|
|
|
|
if(id)
|
|
|
|
|
{
|
|
|
|
|
glDeleteShader(id);
|
|
|
|
|
id = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|