2014-05-08 13:39:12 -04:00
|
|
|
/* RetroArch - A frontend for libretro.
|
2015-01-07 18:06:50 +01:00
|
|
|
* Copyright (C) 2011-2015 - Daniel De Matteis
|
2014-05-08 13:39:12 -04: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 13:42:40 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
2015-06-30 00:38:10 +02:00
|
|
|
#include "video_shader_driver.h"
|
2015-11-23 12:03:38 +01:00
|
|
|
#include "../verbosity.h"
|
2015-06-30 00:38:10 +02:00
|
|
|
|
2014-10-02 13:42:40 +02:00
|
|
|
static const shader_backend_t *shader_ctx_drivers[] = {
|
|
|
|
#ifdef HAVE_GLSL
|
|
|
|
&gl_glsl_backend,
|
|
|
|
#endif
|
2015-02-09 21:49:29 +01:00
|
|
|
#ifdef HAVE_CG
|
|
|
|
&gl_cg_backend,
|
|
|
|
#endif
|
2014-10-02 13:42:40 +02:00
|
|
|
#ifdef HAVE_HLSL
|
|
|
|
&hlsl_backend,
|
|
|
|
#endif
|
|
|
|
&shader_null_backend,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2015-12-05 07:33:21 +01:00
|
|
|
static void *shader_data;
|
|
|
|
|
2015-01-11 20:20:34 +01:00
|
|
|
/**
|
|
|
|
* shader_ctx_find_driver:
|
|
|
|
* @ident : Identifier of shader context driver to find.
|
|
|
|
*
|
|
|
|
* Finds shader context driver and initializes.
|
|
|
|
*
|
|
|
|
* Returns: shader context driver if found, otherwise NULL.
|
|
|
|
**/
|
2014-10-02 13:42:40 +02:00
|
|
|
const shader_backend_t *shader_ctx_find_driver(const char *ident)
|
|
|
|
{
|
|
|
|
unsigned i;
|
2015-01-10 03:04:10 +01:00
|
|
|
|
2014-10-02 13:42:40 +02:00
|
|
|
for (i = 0; shader_ctx_drivers[i]; i++)
|
|
|
|
{
|
2015-01-11 20:20:34 +01:00
|
|
|
if (!strcmp(shader_ctx_drivers[i]->ident, ident))
|
2014-10-02 13:42:40 +02:00
|
|
|
return shader_ctx_drivers[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-11 20:20:34 +01:00
|
|
|
/**
|
|
|
|
* shader_ctx_init_first:
|
|
|
|
*
|
|
|
|
* Finds first suitable shader context driver and initializes.
|
|
|
|
*
|
|
|
|
* Returns: shader context driver if found, otherwise NULL.
|
|
|
|
**/
|
2014-10-02 13:42:40 +02:00
|
|
|
const shader_backend_t *shader_ctx_init_first(void)
|
|
|
|
{
|
|
|
|
unsigned i;
|
2015-01-10 03:04:10 +01:00
|
|
|
|
2014-10-02 13:42:40 +02:00
|
|
|
for (i = 0; shader_ctx_drivers[i]; i++)
|
|
|
|
return shader_ctx_drivers[i];
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
2015-02-14 05:24:20 +01:00
|
|
|
|
|
|
|
struct video_shader *video_shader_driver_get_current_shader(void)
|
|
|
|
{
|
2015-12-06 11:22:16 +01:00
|
|
|
void *video_driver = video_driver_get_ptr(false);
|
2015-11-23 22:14:39 +01:00
|
|
|
const video_poke_interface_t *video_poke = video_driver_get_poke();
|
2015-12-06 11:22:16 +01:00
|
|
|
if (!video_poke || !video_driver)
|
2015-02-14 05:24:20 +01:00
|
|
|
return NULL;
|
2015-11-23 22:14:39 +01:00
|
|
|
if (!video_poke->get_current_shader)
|
2015-02-14 05:24:20 +01:00
|
|
|
return NULL;
|
2015-12-06 11:22:16 +01:00
|
|
|
return video_poke->get_current_shader(video_driver);
|
2015-02-14 05:24:20 +01:00
|
|
|
}
|
2015-10-07 18:18:49 +02:00
|
|
|
|
|
|
|
void video_shader_scale(unsigned idx,
|
|
|
|
const shader_backend_t *shader, struct gfx_fbo_scale *scale)
|
|
|
|
{
|
2015-10-08 06:25:58 +02:00
|
|
|
if (!scale || !shader)
|
2015-10-07 18:18:49 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
scale->valid = false;
|
|
|
|
|
|
|
|
if (shader->shader_scale)
|
2015-12-05 07:33:21 +01:00
|
|
|
shader->shader_scale(shader_data, idx, scale);
|
2015-10-07 18:18:49 +02:00
|
|
|
}
|
2015-12-04 15:50:40 +01:00
|
|
|
|
2015-12-04 15:53:02 +01:00
|
|
|
bool video_shader_driver_init(const shader_backend_t *shader, void *data, const char *path)
|
2015-12-04 15:50:40 +01:00
|
|
|
{
|
2015-12-05 07:33:21 +01:00
|
|
|
void *tmp = NULL;
|
|
|
|
|
2015-12-04 15:50:40 +01:00
|
|
|
if (!shader || !shader->init)
|
|
|
|
return false;
|
2015-12-05 07:33:21 +01:00
|
|
|
|
|
|
|
tmp = shader->init(data, path);
|
|
|
|
|
|
|
|
if (!tmp)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
shader_data = tmp;
|
|
|
|
|
|
|
|
return true;
|
2015-12-04 15:50:40 +01:00
|
|
|
}
|
2015-12-04 15:56:36 +01:00
|
|
|
|
|
|
|
void video_shader_driver_deinit(const shader_backend_t *shader)
|
|
|
|
{
|
|
|
|
if (!shader)
|
|
|
|
return;
|
2015-12-05 07:33:21 +01:00
|
|
|
|
2015-12-06 21:19:35 +01:00
|
|
|
if (shader->deinit)
|
|
|
|
shader->deinit(shader_data);
|
2015-12-05 07:33:21 +01:00
|
|
|
|
|
|
|
shader_data = NULL;
|
2015-12-04 15:56:36 +01:00
|
|
|
}
|
2015-12-04 16:01:09 +01:00
|
|
|
|
|
|
|
void video_shader_driver_use(const shader_backend_t *shader, void *data, unsigned index)
|
|
|
|
{
|
|
|
|
if (!shader)
|
|
|
|
return;
|
2015-12-05 07:33:21 +01:00
|
|
|
shader->use(data, shader_data, index);
|
2015-12-04 16:01:09 +01:00
|
|
|
}
|
2015-12-04 16:05:42 +01:00
|
|
|
|
|
|
|
const char *video_shader_driver_get_ident(const shader_backend_t *shader)
|
|
|
|
{
|
|
|
|
if (!shader)
|
|
|
|
return NULL;
|
|
|
|
return shader->ident;
|
|
|
|
}
|
2015-12-04 16:16:40 +01:00
|
|
|
|
|
|
|
bool video_shader_driver_mipmap_input(const shader_backend_t *shader, unsigned index)
|
|
|
|
{
|
|
|
|
if (!shader)
|
|
|
|
return false;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->mipmap_input(shader_data, index);
|
2015-12-04 16:16:40 +01:00
|
|
|
}
|
2015-12-04 16:19:12 +01:00
|
|
|
|
|
|
|
unsigned video_shader_driver_num_shaders(const shader_backend_t *shader)
|
|
|
|
{
|
|
|
|
if (!shader)
|
|
|
|
return 0;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->num_shaders(shader_data);
|
2015-12-04 16:19:12 +01:00
|
|
|
}
|
2015-12-04 16:33:07 +01:00
|
|
|
|
2015-12-04 16:34:58 +01:00
|
|
|
unsigned video_shader_driver_get_prev_textures(const shader_backend_t *shader)
|
|
|
|
{
|
|
|
|
if (!shader)
|
|
|
|
return 0;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->get_prev_textures(shader_data);
|
2015-12-04 16:34:58 +01:00
|
|
|
}
|
|
|
|
|
2015-12-05 07:33:21 +01:00
|
|
|
bool video_shader_driver_set_coords(const shader_backend_t *shader, void *handle_data, const void *data)
|
2015-12-04 16:33:07 +01:00
|
|
|
{
|
|
|
|
if (!shader || !shader->set_coords)
|
|
|
|
return false;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->set_coords(handle_data, shader_data, data);
|
2015-12-04 16:33:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_set_mvp(const shader_backend_t *shader, void *data, const math_matrix_4x4 *mat)
|
|
|
|
{
|
|
|
|
if (!shader || !shader->set_mvp)
|
|
|
|
return false;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->set_mvp(data, shader_data, mat);
|
2015-12-04 16:33:07 +01:00
|
|
|
}
|
2015-12-04 16:45:38 +01:00
|
|
|
|
|
|
|
bool video_shader_driver_filter_type(const shader_backend_t *shader, unsigned index, bool *smooth)
|
|
|
|
{
|
|
|
|
if (!shader || !shader->filter_type)
|
|
|
|
return false;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->filter_type(shader_data, index, smooth);
|
2015-12-04 16:45:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
enum gfx_wrap_type video_shader_driver_wrap_type(const shader_backend_t *shader, unsigned index)
|
|
|
|
{
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->wrap_type(shader_data, index);
|
2015-12-04 16:45:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool video_shader_driver_get_feedback_pass(const shader_backend_t *shader, unsigned *pass)
|
|
|
|
{
|
|
|
|
if (!shader || !shader->get_feedback_pass)
|
|
|
|
return false;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->get_feedback_pass(shader_data, pass);
|
2015-12-04 16:45:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
struct video_shader *video_shader_driver_direct_get_current_shader(const shader_backend_t *shader)
|
|
|
|
{
|
|
|
|
if (!shader || !shader->get_current_shader)
|
|
|
|
return NULL;
|
2015-12-05 07:33:21 +01:00
|
|
|
return shader->get_current_shader(shader_data);
|
2015-12-04 16:45:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void video_shader_driver_set_params(const shader_backend_t *shader,
|
|
|
|
void *data, unsigned width, unsigned height,
|
|
|
|
unsigned tex_width, unsigned tex_height,
|
|
|
|
unsigned out_width, unsigned out_height,
|
|
|
|
unsigned frame_counter,
|
|
|
|
const void *info,
|
|
|
|
const void *prev_info,
|
|
|
|
const void *feedback_info,
|
|
|
|
const void *fbo_info, unsigned fbo_info_cnt)
|
|
|
|
{
|
|
|
|
if (!shader || !shader->set_params)
|
2015-12-05 07:33:21 +01:00
|
|
|
return;
|
|
|
|
return shader->set_params(data, shader_data,
|
|
|
|
width, height, tex_width, tex_height,
|
2015-12-04 16:45:38 +01:00
|
|
|
out_width, out_height, frame_counter, info, prev_info, feedback_info,
|
|
|
|
fbo_info, fbo_info_cnt);
|
|
|
|
}
|