RetroArch/gfx/shader_cg.c

459 lines
14 KiB
C
Raw Normal View History

2011-01-17 20:54:58 +01:00
/* SSNES - A Super Nintendo Entertainment System (SNES) Emulator frontend for libsnes.
2011-01-23 20:29:28 +01:00
* Copyright (C) 2010-2011 - Hans-Kristian Arntzen
*
* Some code herein may be based on code found in BSNES.
*
* SSNES is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* SSNES 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 for more details.
*
* You should have received a copy of the GNU General Public License along with SSNES.
* If not, see <http://www.gnu.org/licenses/>.
*/
#include "shader_cg.h"
#include <Cg/cg.h>
#include <Cg/cgGL.h>
#include "general.h"
2011-03-06 19:56:35 +01:00
#include <string.h>
2011-05-22 17:02:09 +02:00
#include "strl.h"
#include "conf/config_file.h"
// Used when we call deactivate() since just unbinding the program didn't seem to work... :(
static const char* stock_cg_program =
"void main_vertex"
"("
" float4 position : POSITION,"
" float4 color : COLOR,"
" float2 texCoord : TEXCOORD0,"
""
" uniform float4x4 modelViewProj,"
""
" out float4 oPosition : POSITION,"
" out float4 oColor : COLOR,"
" out float2 otexCoord : TEXCOORD"
")"
"{"
" oPosition = mul(modelViewProj, position);"
" oColor = color;"
" otexCoord = texCoord;"
"}"
""
2011-05-22 17:02:09 +02:00
"float4 main_fragment(float2 tex : TEXCOORD0, uniform sampler2D s0 : TEXUNIT0) : COLOR"
"{"
2011-05-22 17:02:09 +02:00
" return tex2D(s0, tex);"
"}";
static CGcontext cgCtx;
2011-03-06 19:56:35 +01:00
struct cg_program
{
CGprogram vprg;
CGprogram fprg;
CGparameter vid_size_f;
CGparameter tex_size_f;
CGparameter out_size_f;
CGparameter frame_cnt_f;
2011-03-06 19:56:35 +01:00
CGparameter vid_size_v;
CGparameter tex_size_v;
CGparameter out_size_v;
CGparameter frame_cnt_v;
2011-03-06 19:56:35 +01:00
CGparameter mvp;
};
2011-05-22 17:02:09 +02:00
#define FILTER_UNSPEC 0
#define FILTER_LINEAR 1
#define FILTER_NEAREST 2
#define MAX_SHADERS 16
static struct cg_program prg[MAX_SHADERS];
static bool cg_active = false;
2011-03-06 19:56:35 +01:00
static CGprofile cgVProf, cgFProf;
static unsigned active_index = 0;
2011-05-22 17:02:09 +02:00
static unsigned cg_shader_num = 0;
static struct gl_fbo_scale cg_scale[MAX_SHADERS];
static unsigned fbo_smooth[MAX_SHADERS];
void gl_cg_set_proj_matrix(void)
{
if (cg_active)
2011-03-06 19:56:35 +01:00
cgGLSetStateMatrixParameter(prg[active_index].mvp, CG_GL_MODELVIEW_PROJECTION_MATRIX, CG_GL_MATRIX_IDENTITY);
}
void gl_cg_set_params(unsigned width, unsigned height,
unsigned tex_width, unsigned tex_height,
unsigned out_width, unsigned out_height,
unsigned frame_count)
{
if (cg_active)
{
2011-03-06 19:56:35 +01:00
cgGLSetParameter2f(prg[active_index].vid_size_f, width, height);
cgGLSetParameter2f(prg[active_index].tex_size_f, tex_width, tex_height);
cgGLSetParameter2f(prg[active_index].out_size_f, out_width, out_height);
cgGLSetParameter1f(prg[active_index].frame_cnt_f, (float)frame_count);
2011-03-06 19:56:35 +01:00
cgGLSetParameter2f(prg[active_index].vid_size_v, width, height);
cgGLSetParameter2f(prg[active_index].tex_size_v, tex_width, tex_height);
cgGLSetParameter2f(prg[active_index].out_size_v, out_width, out_height);
cgGLSetParameter1f(prg[active_index].frame_cnt_v, (float)frame_count);
}
}
void gl_cg_deinit(void)
{
if (cg_active)
cgDestroyContext(cgCtx);
2011-03-29 19:18:06 +02:00
cg_active = false;
2011-05-22 17:02:09 +02:00
cg_shader_num = 0;
memset(prg, 0, sizeof(prg));
memset(cg_scale, 0, sizeof(cg_scale));
memset(fbo_smooth, 0, sizeof(fbo_smooth));
}
2011-05-22 17:02:09 +02:00
static bool load_plain(const char *path)
{
SSNES_LOG("Loading Cg file: %s\n", path);
2011-03-06 19:56:35 +01:00
if (strlen(g_settings.video.second_pass_shader) > 0)
SSNES_LOG("Loading 2nd pass: %s\n", g_settings.video.second_pass_shader);
prg[0].fprg = cgCreateProgram(cgCtx, CG_SOURCE, stock_cg_program, cgFProf, "main_fragment", 0);
prg[0].vprg = cgCreateProgram(cgCtx, CG_SOURCE, stock_cg_program, cgVProf, "main_vertex", 0);
prg[1].fprg = cgCreateProgramFromFile(cgCtx, CG_SOURCE, path, cgFProf, "main_fragment", 0);
prg[1].vprg = cgCreateProgramFromFile(cgCtx, CG_SOURCE, path, cgVProf, "main_vertex", 0);
if (strlen(g_settings.video.second_pass_shader) > 0)
{
2011-03-06 19:56:35 +01:00
prg[2].fprg = cgCreateProgramFromFile(cgCtx, CG_SOURCE, g_settings.video.second_pass_shader, cgFProf, "main_fragment", 0);
prg[2].vprg = cgCreateProgramFromFile(cgCtx, CG_SOURCE, g_settings.video.second_pass_shader, cgVProf, "main_vertex", 0);
2011-05-22 17:02:09 +02:00
cg_shader_num = 2;
2011-03-06 19:56:35 +01:00
}
else
2011-05-22 17:02:09 +02:00
{
2011-03-06 19:56:35 +01:00
prg[2] = prg[0];
2011-05-22 17:02:09 +02:00
cg_shader_num = 1;
}
2011-03-06 19:56:35 +01:00
for (int i = 0; i < 3; i++)
{
2011-05-22 17:36:18 +02:00
if (!prg[i].fprg || !prg[i].vprg)
2011-03-06 19:56:35 +01:00
{
CGerror err = cgGetError();
SSNES_ERR("CG error: %s\n", cgGetErrorString(err));
return false;
}
cgGLLoadProgram(prg[i].fprg);
cgGLLoadProgram(prg[i].vprg);
}
2011-03-06 19:56:35 +01:00
2011-05-22 17:02:09 +02:00
return true;
}
static bool load_preset(const char *path)
{
// Create passthrough shader.
prg[0].fprg = cgCreateProgram(cgCtx, CG_SOURCE, stock_cg_program, cgFProf, "main_fragment", 0);
prg[0].vprg = cgCreateProgram(cgCtx, CG_SOURCE, stock_cg_program, cgVProf, "main_vertex", 0);
if (!prg[0].fprg || !prg[0].vprg)
{
SSNES_ERR("Failed to compile passthrough shader, is something wrong with your environment?\n");
return false;
}
SSNES_LOG("Loading Cg meta-shader: %s\n", path);
config_file_t *conf = config_file_new(path);
if (!conf)
{
SSNES_ERR("Failed to load preset.\n");
goto error;
}
int shaders;
2011-05-22 17:07:47 +02:00
if (!config_get_int(conf, "shaders", &shaders))
2011-05-22 17:02:09 +02:00
{
SSNES_ERR("Cannot find \"shaders\" param.\n");
goto error;
}
if (shaders < 1)
{
SSNES_ERR("Need to define at least 1 shader!\n");
goto error;
}
cg_shader_num = shaders;
// Check filter params.
for (unsigned i = 0; i < shaders; i++)
{
bool smooth;
char filter_name_buf[64];
snprintf(filter_name_buf, sizeof(filter_name_buf), "filter_linear%u", i);
if (config_get_bool(conf, filter_name_buf, &smooth))
fbo_smooth[i + 1] = smooth ? FILTER_LINEAR : FILTER_NEAREST;
}
// Bigass for-loop ftw. Check scaling params.
for (unsigned i = 0; i < shaders; i++)
{
char *scale_type;
char scale_name_buf[64];
snprintf(scale_name_buf, sizeof(scale_name_buf), "scale_type%u", i);
if (config_get_string(conf, scale_name_buf, &scale_type))
{
char attr_name_buf[64];
double fattr;
int iattr;
struct gl_fbo_scale *scale = &cg_scale[i + 1]; // Shader 0 is passthrough shader. Start at 1.
scale->valid = true;
scale->type_x = SSNES_SCALE_INPUT;
scale->type_y = SSNES_SCALE_INPUT;
scale->scale_x = SSNES_SCALE_INPUT;
scale->scale_y = SSNES_SCALE_INPUT;
// Check source scale.
if (strcmp(scale_type, "source") == 0)
{
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale%u", i);
if (config_get_double(conf, attr_name_buf, &fattr))
{
scale->scale_x = fattr;
scale->scale_y = fattr;
}
else
{
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale_x%u", i);
if (config_get_double(conf, attr_name_buf, &fattr))
scale->scale_x = fattr;
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale_y%u", i);
if (config_get_double(conf, attr_name_buf, &fattr))
scale->scale_y = fattr;
}
}
// Viewport scale.
else if (strcmp(scale_type, "viewport") == 0)
{
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale%u", i);
if (config_get_double(conf, attr_name_buf, &fattr))
{
scale->scale_x = fattr;
scale->scale_y = fattr;
scale->type_x = SSNES_SCALE_VIEWPORT;
scale->type_y = SSNES_SCALE_VIEWPORT;
}
else
{
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale_x%u", i);
if (config_get_double(conf, attr_name_buf, &fattr))
{
scale->scale_x = fattr;
scale->type_x = SSNES_SCALE_VIEWPORT;
}
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale_y%u", i);
if (config_get_double(conf, attr_name_buf, &fattr))
{
scale->scale_y = fattr;
scale->type_y = SSNES_SCALE_VIEWPORT;
}
}
}
// Absolute pixel scale.
else if (strcmp(scale_type, "absolute") == 0)
{
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale%u", i);
if (config_get_int(conf, attr_name_buf, &iattr))
{
scale->abs_x = iattr;
scale->abs_y = iattr;
scale->type_x = SSNES_SCALE_ABSOLUTE;
scale->type_y = SSNES_SCALE_ABSOLUTE;
}
else
{
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale_x%u", i);
if (config_get_int(conf, attr_name_buf, &iattr))
{
scale->abs_x = iattr;
scale->type_x = SSNES_SCALE_ABSOLUTE;
}
snprintf(attr_name_buf, sizeof(attr_name_buf), "scale_y%u", i);
if (config_get_int(conf, attr_name_buf, &iattr))
{
scale->abs_y = iattr;
scale->type_y = SSNES_SCALE_ABSOLUTE;
}
}
}
else
{
SSNES_ERR("Invalid attribute: \"%s\"\n", scale_type);
free(scale_type);
goto error;
}
free(scale_type);
}
}
2011-05-22 17:36:18 +02:00
// Basedir.
2011-05-22 17:07:47 +02:00
char dir_path[256];
strlcpy(dir_path, path, sizeof(dir_path));
char *ptr = strrchr(dir_path, '/');
if (!ptr) ptr = strrchr(dir_path, '\\');
2011-05-22 17:36:18 +02:00
if (ptr)
ptr[1] = '\0';
else // No directory.
dir_path[0] = '\0';
2011-05-22 17:07:47 +02:00
2011-05-22 17:02:09 +02:00
// Finally load shaders :)
for (unsigned i = 0; i < shaders; i++)
{
char *shader_path;
char attr_buf[64];
char path_buf[512];
snprintf(attr_buf, sizeof(attr_buf), "shader%u", i);
if (config_get_string(conf, attr_buf, &shader_path))
{
2011-05-22 17:07:47 +02:00
strlcpy(path_buf, dir_path, sizeof(path_buf));
2011-05-22 17:02:09 +02:00
strlcat(path_buf, shader_path, sizeof(path_buf));
free(shader_path);
}
else
{
SSNES_ERR("Didn't find shader path in config ...\n");
goto error;
}
SSNES_LOG("Loading Cg shader: \"%s\".\n", path_buf);
struct cg_program *prog = &prg[i + 1];
prog->fprg = cgCreateProgramFromFile(cgCtx, CG_SOURCE, path_buf, cgFProf, "main_fragment", 0);
prog->vprg = cgCreateProgramFromFile(cgCtx, CG_SOURCE, path_buf, cgVProf, "main_vertex", 0);
if (!prog->fprg || !prog->vprg)
{
CGerror err = cgGetError();
SSNES_ERR("CG error: %s\n", cgGetErrorString(err));
goto error;
}
cgGLLoadProgram(prog->fprg);
cgGLLoadProgram(prog->vprg);
}
// TODO: Load textures ...
config_file_free(conf);
return true;
error:
if (conf)
config_file_free(conf);
return false;
}
bool gl_cg_init(const char *path)
{
cgCtx = cgCreateContext();
if (cgCtx == NULL)
{
SSNES_ERR("Failed to create Cg context\n");
return false;
}
cgFProf = cgGLGetLatestProfile(CG_GL_FRAGMENT);
cgVProf = cgGLGetLatestProfile(CG_GL_VERTEX);
if (cgFProf == CG_PROFILE_UNKNOWN || cgVProf == CG_PROFILE_UNKNOWN)
{
SSNES_ERR("Invalid profile type\n");
return false;
}
cgGLSetOptimalOptions(cgFProf);
cgGLSetOptimalOptions(cgVProf);
2011-05-22 17:36:18 +02:00
cgGLEnableProfile(cgFProf);
cgGLEnableProfile(cgVProf);
2011-05-22 17:02:09 +02:00
if (strstr(path, ".cgp"))
{
if (!load_preset(path))
return false;
}
else
{
if (!load_plain(path))
return false;
}
2011-05-22 17:36:18 +02:00
for (unsigned i = 1; i < cg_shader_num + 1; i++)
{
2011-03-06 19:56:35 +01:00
cgGLBindProgram(prg[i].fprg);
cgGLBindProgram(prg[i].vprg);
prg[i].vid_size_f = cgGetNamedParameter(prg[i].fprg, "IN.video_size");
prg[i].tex_size_f = cgGetNamedParameter(prg[i].fprg, "IN.texture_size");
prg[i].out_size_f = cgGetNamedParameter(prg[i].fprg, "IN.output_size");
prg[i].frame_cnt_f = cgGetNamedParameter(prg[i].fprg, "IN.frame_count");
2011-03-06 19:56:35 +01:00
prg[i].vid_size_v = cgGetNamedParameter(prg[i].vprg, "IN.video_size");
prg[i].tex_size_v = cgGetNamedParameter(prg[i].vprg, "IN.texture_size");
prg[i].out_size_v = cgGetNamedParameter(prg[i].vprg, "IN.output_size");
2011-05-22 17:02:09 +02:00
prg[i].frame_cnt_v = cgGetNamedParameter(prg[i].vprg, "IN.frame_count");
2011-03-06 19:56:35 +01:00
prg[i].mvp = cgGetNamedParameter(prg[i].vprg, "modelViewProj");
cgGLSetStateMatrixParameter(prg[i].mvp, CG_GL_MODELVIEW_PROJECTION_MATRIX, CG_GL_MATRIX_IDENTITY);
}
2011-03-06 19:56:35 +01:00
cgGLBindProgram(prg[1].fprg);
cgGLBindProgram(prg[1].vprg);
cg_active = true;
return true;
}
2011-03-06 19:56:35 +01:00
void gl_cg_use(unsigned index)
{
2011-05-22 17:02:09 +02:00
if (cg_active && prg[index].vprg && prg[index].fprg)
{
2011-03-06 19:56:35 +01:00
active_index = index;
cgGLBindProgram(prg[index].vprg);
cgGLBindProgram(prg[index].fprg);
}
}
2011-03-12 15:30:57 +01:00
unsigned gl_cg_num(void)
{
if (cg_active)
2011-05-22 17:02:09 +02:00
return cg_shader_num;
2011-03-12 15:30:57 +01:00
else
return 0;
}
2011-03-14 21:28:30 +01:00
bool gl_cg_filter_type(unsigned index, bool *smooth)
{
2011-05-22 17:02:09 +02:00
if (cg_active)
{
if (fbo_smooth[index] == FILTER_UNSPEC)
return false;
*smooth = (fbo_smooth[index] == FILTER_LINEAR);
return true;
}
else
return false;
2011-03-14 21:28:30 +01:00
}
void gl_cg_shader_scale(unsigned index, struct gl_fbo_scale *scale)
{
2011-05-22 17:02:09 +02:00
if (cg_active)
*scale = cg_scale[index];
else
scale->valid = false;
}