2014-05-08 17:39:12 +00:00
|
|
|
/* RetroArch - A frontend for libretro.
|
2016-01-10 03:41:52 +00:00
|
|
|
* Copyright (C) 2011-2016 - Daniel De Matteis
|
2014-05-08 17:39:12 +00:00
|
|
|
*
|
|
|
|
* RetroArch 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.
|
|
|
|
*
|
|
|
|
* RetroArch 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 RetroArch.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2014-10-02 11:42:40 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
2016-01-20 03:14:55 +00:00
|
|
|
#include <string/stdstring.h>
|
|
|
|
|
2015-06-29 22:38:10 +00:00
|
|
|
#include "video_shader_driver.h"
|
2015-11-23 11:03:38 +00:00
|
|
|
#include "../verbosity.h"
|
2015-06-29 22:38:10 +00:00
|
|
|
|
2014-10-02 11:42:40 +00:00
|
|
|
static const shader_backend_t *shader_ctx_drivers[] = {
|
|
|
|
#ifdef HAVE_GLSL
|
|
|
|
&gl_glsl_backend,
|
|
|
|
#endif
|
2015-02-09 20:49:29 +00:00
|
|
|
#ifdef HAVE_CG
|
|
|
|
&gl_cg_backend,
|
|
|
|
#endif
|
2014-10-02 11:42:40 +00:00
|
|
|
#ifdef HAVE_HLSL
|
|
|
|
&hlsl_backend,
|
|
|
|
#endif
|
|
|
|
&shader_null_backend,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
static const shader_backend_t *current_shader = NULL;
|
|
|
|
static void *shader_data = NULL;
|
|
|
|
|
2016-05-05 03:53:50 +00:00
|
|
|
static const shader_backend_t *video_shader_set_backend(enum rarch_shader_type type)
|
2016-04-16 03:20:41 +00:00
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case RARCH_SHADER_CG:
|
2016-04-16 03:36:09 +00:00
|
|
|
{
|
2016-05-05 03:52:38 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_CG
|
|
|
|
gfx_ctx_flags_t flags;
|
2016-05-08 18:32:46 +00:00
|
|
|
video_context_driver_get_flags(&flags);
|
2016-05-05 03:58:40 +00:00
|
|
|
|
2016-05-05 15:52:30 +00:00
|
|
|
if (BIT32_GET(flags.flags, GFX_CTX_FLAGS_GL_CORE_CONTEXT))
|
2016-05-05 03:52:38 +00:00
|
|
|
{
|
|
|
|
RARCH_ERR("[Shader driver]: Cg cannot be used with core GL context. Trying to fall back to GLSL...\n");
|
2016-05-05 15:44:10 +00:00
|
|
|
return video_shader_set_backend(RARCH_SHADER_GLSL);
|
2016-05-05 03:52:38 +00:00
|
|
|
}
|
2016-05-05 15:44:43 +00:00
|
|
|
|
|
|
|
RARCH_LOG("[Shader driver]: Using Cg shader backend.\n");
|
|
|
|
return &gl_cg_backend;
|
2016-04-16 03:20:41 +00:00
|
|
|
#else
|
2016-05-05 03:52:38 +00:00
|
|
|
break;
|
2016-04-16 03:20:41 +00:00
|
|
|
#endif
|
2016-05-05 03:52:38 +00:00
|
|
|
}
|
2016-04-16 03:20:41 +00:00
|
|
|
case RARCH_SHADER_GLSL:
|
|
|
|
#ifdef HAVE_GLSL
|
|
|
|
RARCH_LOG("[Shader driver]: Using GLSL shader backend.\n");
|
|
|
|
return &gl_glsl_backend;
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case RARCH_SHADER_NONE:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
bool video_shader_driver_get_prev_textures(video_shader_ctx_texture_t *texture)
|
2016-02-14 17:12:42 +00:00
|
|
|
{
|
2016-05-08 19:11:27 +00:00
|
|
|
if (!!texture || !current_shader)
|
2016-02-14 17:12:42 +00:00
|
|
|
{
|
2016-05-23 20:30:01 +00:00
|
|
|
if (texture)
|
|
|
|
texture->id = 0;
|
2016-05-08 19:11:27 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
texture->id = current_shader->get_prev_textures(shader_data);
|
2016-02-14 20:55:19 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2016-02-14 20:55:19 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
bool video_shader_driver_get_ident(video_shader_ctx_ident_t *ident)
|
|
|
|
{
|
|
|
|
if (!current_shader || !ident)
|
|
|
|
return false;
|
|
|
|
ident->ident = current_shader->ident;
|
|
|
|
return true;
|
|
|
|
}
|
2016-02-14 17:22:38 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
bool video_shader_driver_get_current_shader(video_shader_ctx_t *shader)
|
|
|
|
{
|
|
|
|
void *video_driver = video_driver_get_ptr(false);
|
|
|
|
const video_poke_interface_t *video_poke = video_driver_get_poke();
|
2016-02-14 17:22:38 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
shader->data = NULL;
|
|
|
|
if (!video_poke || !video_driver)
|
|
|
|
return false;
|
|
|
|
if (!video_poke->get_current_shader)
|
|
|
|
return false;
|
|
|
|
shader->data = video_poke->get_current_shader(video_driver);
|
|
|
|
return true;
|
|
|
|
}
|
2016-04-16 19:04:13 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
bool video_shader_driver_direct_get_current_shader(video_shader_ctx_t *shader)
|
|
|
|
{
|
|
|
|
shader->data = NULL;
|
|
|
|
if (!current_shader || !current_shader->get_current_shader)
|
|
|
|
return false;
|
2016-02-14 17:17:43 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
shader->data = current_shader->get_current_shader(shader_data);
|
|
|
|
return true;
|
|
|
|
}
|
2016-02-14 17:41:45 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
bool video_shader_driver_deinit(void)
|
|
|
|
{
|
|
|
|
if (!current_shader)
|
|
|
|
return false;
|
2016-02-14 17:41:45 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
if (current_shader->deinit)
|
|
|
|
current_shader->deinit(shader_data);
|
2016-02-14 17:41:45 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
shader_data = NULL;
|
|
|
|
current_shader = NULL;
|
|
|
|
return true;
|
|
|
|
}
|
2016-02-14 17:41:45 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
bool video_shader_driver_set_parameter(struct uniform_info *param)
|
|
|
|
{
|
|
|
|
if (!current_shader || !param)
|
|
|
|
return false;
|
2016-05-15 15:26:23 +00:00
|
|
|
current_shader->set_uniform_parameter(shader_data,
|
|
|
|
param, NULL);
|
2016-05-08 19:11:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
2016-02-14 18:21:54 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
bool video_shader_driver_set_parameters(video_shader_ctx_params_t *params)
|
|
|
|
{
|
|
|
|
if (!current_shader || !current_shader->set_params)
|
|
|
|
return false;
|
|
|
|
current_shader->set_params(
|
|
|
|
params->data,
|
|
|
|
shader_data,
|
|
|
|
params->width,
|
|
|
|
params->height,
|
|
|
|
params->tex_width,
|
|
|
|
params->tex_height,
|
|
|
|
params->out_width,
|
|
|
|
params->out_height,
|
|
|
|
params->frame_counter,
|
|
|
|
params->info,
|
|
|
|
params->prev_info,
|
|
|
|
params->feedback_info,
|
|
|
|
params->fbo_info,
|
|
|
|
params->fbo_info_cnt);
|
|
|
|
return true;
|
|
|
|
}
|
2016-02-14 18:21:54 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
/* Finds first suitable shader context driver. */
|
|
|
|
bool video_shader_driver_init_first(void)
|
|
|
|
{
|
|
|
|
unsigned i;
|
2016-02-14 18:21:54 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
for (i = 0; shader_ctx_drivers[i]; i++)
|
|
|
|
{
|
|
|
|
current_shader = shader_ctx_drivers[i];
|
|
|
|
return true;
|
2016-02-14 17:12:42 +00:00
|
|
|
}
|
2016-05-08 19:11:27 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_init(video_shader_ctx_init_t *init)
|
|
|
|
{
|
|
|
|
void *tmp = NULL;
|
2016-02-14 17:12:42 +00:00
|
|
|
|
2016-05-08 19:11:27 +00:00
|
|
|
if (!init->shader || !init->shader->init)
|
|
|
|
{
|
|
|
|
init->shader = video_shader_set_backend(init->shader_type);
|
|
|
|
|
|
|
|
if (!init->shader)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = init->shader->init(init->data, init->path);
|
|
|
|
|
|
|
|
if (!tmp)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
shader_data = tmp;
|
|
|
|
current_shader = init->shader;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_get_feedback_pass(unsigned *data)
|
|
|
|
{
|
|
|
|
if (!current_shader || !current_shader->get_feedback_pass)
|
|
|
|
return false;
|
|
|
|
if (!current_shader->get_feedback_pass(shader_data, data))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_mipmap_input(unsigned *index)
|
|
|
|
{
|
|
|
|
if (!current_shader)
|
|
|
|
return false;
|
|
|
|
if (!current_shader->mipmap_input(shader_data, *index))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_set_coords(video_shader_ctx_coords_t *coords)
|
|
|
|
{
|
|
|
|
if (!current_shader || !current_shader->set_coords)
|
|
|
|
return false;
|
|
|
|
if (!current_shader->set_coords(coords->handle_data,
|
2016-05-10 00:32:49 +00:00
|
|
|
shader_data, (const struct video_coords*)coords->data))
|
2016-05-08 19:11:27 +00:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_scale(video_shader_ctx_scale_t *scaler)
|
|
|
|
{
|
|
|
|
if (!scaler || !scaler->scale)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
scaler->scale->valid = false;
|
|
|
|
|
|
|
|
if (!current_shader || !current_shader->shader_scale)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
current_shader->shader_scale(shader_data, scaler->idx, scaler->scale);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_info(video_shader_ctx_info_t *shader_info)
|
|
|
|
{
|
|
|
|
if (!shader_info || !current_shader)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
shader_info->num = 0;
|
|
|
|
if (current_shader->num_shaders)
|
|
|
|
shader_info->num = current_shader->num_shaders(shader_data);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_set_mvp(video_shader_ctx_mvp_t *mvp)
|
|
|
|
{
|
|
|
|
if (!current_shader || !current_shader->set_mvp)
|
|
|
|
return false;
|
|
|
|
if (!mvp || !mvp->matrix)
|
|
|
|
return false;
|
|
|
|
if (!current_shader->set_mvp(mvp->data, shader_data, mvp->matrix))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_filter_type(video_shader_ctx_filter_t *filter)
|
|
|
|
{
|
|
|
|
if (!current_shader || !current_shader->filter_type || !filter)
|
|
|
|
return false;
|
|
|
|
if (!current_shader->filter_type(shader_data,
|
|
|
|
filter->index, filter->smooth))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_compile_program(struct shader_program_info *program_info)
|
|
|
|
{
|
|
|
|
if (!current_shader || !program_info)
|
|
|
|
return false;
|
|
|
|
return current_shader->compile_program(program_info->data,
|
|
|
|
program_info->idx, NULL, program_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_use(video_shader_ctx_info_t *shader_info)
|
|
|
|
{
|
|
|
|
if (!current_shader || !shader_info)
|
|
|
|
return false;
|
|
|
|
current_shader->use(shader_info->data, shader_data, shader_info->idx, shader_info->set_active);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_wrap_type(video_shader_ctx_wrap_t *wrap)
|
|
|
|
{
|
|
|
|
if (!current_shader || !current_shader->wrap_type)
|
|
|
|
return false;
|
|
|
|
wrap->type = current_shader->wrap_type(shader_data, wrap->idx);
|
2016-02-14 17:12:42 +00:00
|
|
|
return true;
|
|
|
|
}
|