2012-11-01 15:19:01 +00:00
|
|
|
// Copyright (c) 2012- PPSSPP Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
2012-11-04 22:01:49 +00:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
2012-11-26 03:25:14 +00:00
|
|
|
#if defined(USING_GLES2)
|
2012-11-01 15:19:01 +00:00
|
|
|
#define GLSL_ES_1_0
|
|
|
|
#else
|
|
|
|
#define GLSL_1_3
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#include "FragmentShaderGenerator.h"
|
|
|
|
#include "../ge_constants.h"
|
|
|
|
#include "../GPUState.h"
|
2012-11-03 02:33:24 +00:00
|
|
|
#include <cstdio>
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
// TODO: remove
|
|
|
|
static char buffer[16384];
|
|
|
|
|
|
|
|
// SDL 1.2 on Apple does not have support for OpenGL 3 and hence needs
|
|
|
|
// special treatment in the shader generator.
|
|
|
|
#ifdef __APPLE__
|
|
|
|
#define FORCE_OPENGL_2_0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#define WRITE p+=sprintf
|
|
|
|
|
|
|
|
// GL_NV_shader_framebuffer_fetch looks interesting....
|
|
|
|
|
|
|
|
// Here we must take all the bits of the gstate that determine what the fragment shader will
|
|
|
|
// look like, and concatenate them together into an ID.
|
|
|
|
void ComputeFragmentShaderID(FragmentShaderID *id)
|
|
|
|
{
|
|
|
|
memset(&id->d[0], 0, sizeof(id->d));
|
|
|
|
if (gstate.clearmode & 1) {
|
|
|
|
// We only need one clear shader, so let's ignore the rest of the bits.
|
|
|
|
id->d[0] = 1;
|
|
|
|
} else {
|
|
|
|
// id->d[0] |= (gstate.clearmode & 1);
|
|
|
|
id->d[0] |= (gstate.texfunc & 0x7) << 1;
|
|
|
|
id->d[0] |= ((gstate.texfunc & 0x100) >> 8) << 4; // rgb or rgba
|
|
|
|
id->d[0] |= ((gstate.texfunc & 0x10000) >> 16) << 5; // color double
|
|
|
|
id->d[0] |= (gstate.lmode & 1) << 6;
|
|
|
|
id->d[0] |= (gstate.textureMapEnable & 1) << 7;
|
|
|
|
id->d[0] |= (gstate.alphaTestEnable & 1) << 8;
|
|
|
|
id->d[0] |= (gstate.alphatest & 0x7) << 9; // alpha test func
|
2012-11-28 12:45:22 +00:00
|
|
|
id->d[0] |= (gstate.fogEnable & 1) << 9;
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Missing: Alpha test, color test, Z depth range, fog
|
|
|
|
// Also, logic ops etc, of course. Urgh.
|
2012-11-28 12:45:22 +00:00
|
|
|
// We could do all this with booleans, but I don't trust the shader compilers on
|
|
|
|
// Android devices to be anything but stupid.
|
2012-11-01 15:19:01 +00:00
|
|
|
char *GenerateFragmentShader()
|
|
|
|
{
|
|
|
|
char *p = buffer;
|
|
|
|
#if defined(GLSL_ES_1_0)
|
|
|
|
WRITE(p, "precision mediump float;\n");
|
|
|
|
#elif !defined(FORCE_OPENGL_2_0)
|
|
|
|
WRITE(p, "#version 130\n");
|
|
|
|
#endif
|
|
|
|
|
2012-11-09 16:51:45 +00:00
|
|
|
int lmode = gstate.lmode & 1;
|
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
if (gstate.textureMapEnable & 1)
|
|
|
|
WRITE(p, "uniform sampler2D tex;\n");
|
|
|
|
if (gstate.alphaTestEnable & 1)
|
|
|
|
WRITE(p, "uniform vec4 u_alpharef;\n");
|
2012-11-28 12:45:22 +00:00
|
|
|
if (gstate.fogEnable & 1) {
|
|
|
|
WRITE(p, "uniform vec3 u_fogcolor;\n");
|
|
|
|
WRITE(p, "uniform vec2 u_fogcoef;\n");
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
WRITE(p, "uniform vec4 u_texenv;\n");
|
|
|
|
WRITE(p, "varying vec4 v_color0;\n");
|
2012-11-09 16:51:45 +00:00
|
|
|
if (lmode)
|
2012-11-01 15:19:01 +00:00
|
|
|
WRITE(p, "varying vec4 v_color1;\n");
|
|
|
|
WRITE(p, "varying vec2 v_texcoord;\n");
|
2012-11-28 12:45:22 +00:00
|
|
|
if (gstate.isFogEnabled())
|
|
|
|
WRITE(p, "varying float v_depth;\n");
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2012-11-23 11:43:31 +00:00
|
|
|
WRITE(p, "void main() {\n");
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " vec4 v;\n");
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
if (gstate.clearmode & 1)
|
|
|
|
{
|
2012-11-23 11:43:31 +00:00
|
|
|
WRITE(p, " v = v_color0;\n");
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const char *secondary = "";
|
2012-11-23 00:26:12 +00:00
|
|
|
// Secondary color for specular on top of texture
|
2012-11-09 16:51:45 +00:00
|
|
|
if (lmode) {
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " vec4 s = vec4(v_color1.xyz, 0.0);");
|
2012-11-01 15:19:01 +00:00
|
|
|
secondary = " + s";
|
2012-11-09 16:51:45 +00:00
|
|
|
} else {
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " vec4 s = vec4(0.0, 0.0, 0.0, 0.0);\n");
|
2012-11-09 16:51:45 +00:00
|
|
|
secondary = "";
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (gstate.textureMapEnable & 1) {
|
|
|
|
WRITE(p, " vec4 t = texture2D(tex, v_texcoord);\n");
|
|
|
|
WRITE(p, " vec4 p = clamp(v_color0, 0.0, 1.0);\n");
|
|
|
|
} else {
|
|
|
|
// No texture mapping
|
2012-11-28 12:45:22 +00:00
|
|
|
WRITE(p, " vec4 t = vec4(1.0, 1.0, 1.0, 1.0);\n");
|
|
|
|
WRITE(p, " vec4 p = clamp(v_color0, 0.0, 1.0);\n");
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (gstate.texfunc & 0x100) { // texfmt == RGBA
|
|
|
|
switch (gstate.texfunc & 0x7) {
|
|
|
|
case GE_TEXFUNC_MODULATE:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = t * p%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_DECAL:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4(1.0 - t.a * p.rgb + t.a * u_texenv.rgb, p.a)%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_BLEND:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4((1.0 - t.rgb) * p.rgb + t.rgb * u_texenv.rgb, p.a * t.a)%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_REPLACE:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = t%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_ADD:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4(t.rgb + p.rgb, p.a * t.a)%s;\n", secondary); break;
|
2012-11-23 11:43:31 +00:00
|
|
|
default:
|
|
|
|
WRITE(p, " v = p;\n"); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
} else { // texfmt == RGB
|
|
|
|
switch (gstate.texfunc & 0x7) {
|
|
|
|
case GE_TEXFUNC_MODULATE:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4(t.rgb * p.rgb, p.a)%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_DECAL:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4(t.rgb, p.a)%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_BLEND:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4(1.0 - t.rgb) * p.rgb + t.rgb * u_texenv.rgb, p.a)%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_REPLACE:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4(t.rgb, p.a)%s;\n", secondary); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
case GE_TEXFUNC_ADD:
|
2012-11-23 00:26:12 +00:00
|
|
|
WRITE(p, " v = vec4(t.rgb + p.rgb, p.a)%s;\n", secondary); break;
|
2012-11-23 11:43:31 +00:00
|
|
|
default:
|
|
|
|
WRITE(p, " v = p;\n"); break;
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-23 00:26:12 +00:00
|
|
|
// Color doubling
|
|
|
|
if (gstate.texfunc & 0x10000) {
|
2012-11-23 11:43:31 +00:00
|
|
|
WRITE(p, " v = v * vec4(2.0, 2.0, 2.0, 1.0);");
|
2012-11-23 00:26:12 +00:00
|
|
|
}
|
2012-11-23 11:43:31 +00:00
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
if (gstate.alphaTestEnable & 1) {
|
|
|
|
int alphaTestFunc = gstate.alphatest & 7;
|
|
|
|
const char *alphaTestFuncs[] = { "#", "#", " == ", " != ", " < ", " <= ", " > ", " >= " }; // never/always don't make sense
|
2012-11-23 11:43:31 +00:00
|
|
|
if (alphaTestFuncs[alphaTestFunc][0] != '#')
|
|
|
|
WRITE(p, "if (!(v.a %s u_alpharef.x)) discard;", alphaTestFuncs[alphaTestFunc]);
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2012-11-28 12:45:22 +00:00
|
|
|
if (gstate.isFogEnabled()) {
|
|
|
|
// Haven't figured out how to adjust the depth range yet.
|
|
|
|
// WRITE(p, " v = mix(v, u_fogcolor, u_fogcoef.x + u_fogcoef.y * v_depth;\n");
|
|
|
|
// WRITE(p, " v.x = v_depth;\n");
|
|
|
|
}
|
|
|
|
// Fogging should be added here - and disabled during clear mode
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
2012-11-23 11:43:31 +00:00
|
|
|
|
|
|
|
WRITE(p, " gl_FragColor = v;\n");
|
2012-11-01 15:19:01 +00:00
|
|
|
WRITE(p, "}\n");
|
|
|
|
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|