RetroArch/gfx/font_driver.c

389 lines
9.4 KiB
C
Raw Normal View History

2012-04-21 21:13:50 +00:00
/* RetroArch - A frontend for libretro.
2014-01-01 00:50:59 +00:00
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2016-01-10 03:41:52 +00:00
* Copyright (C) 2011-2016 - Daniel De Matteis
2012-02-12 14:23:35 +00:00
*
2012-04-21 21:13:50 +00:00
* RetroArch is free software: you can redistribute it and/or modify it under the terms
2012-02-12 14:23:35 +00:00
* 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.
*
2012-04-21 21:13:50 +00:00
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
2012-02-12 14:23:35 +00:00
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
2012-04-21 21:31:57 +00:00
* You should have received a copy of the GNU General Public License along with RetroArch.
2012-02-12 14:23:35 +00:00
* If not, see <http://www.gnu.org/licenses/>.
*/
#include "font_driver.h"
2015-12-05 10:29:06 +00:00
#include "video_thread_wrapper.h"
2016-09-01 03:38:26 +00:00
#include "../configuration.h"
#include "../verbosity.h"
2016-09-06 21:47:05 +00:00
#ifdef HAVE_CONFIG_H
#include "../config.h"
#endif
#include <stdlib.h>
static const font_renderer_driver_t *font_backends[] = {
#ifdef HAVE_FREETYPE
&freetype_font_renderer,
#endif
#if defined(__APPLE__) && defined(HAVE_CORETEXT)
&coretext_font_renderer,
#endif
#ifdef HAVE_STB_FONT
#if defined(VITA) || defined(ANDROID) || defined(_WIN32) && !defined(_XBOX)
2016-11-05 15:20:18 +00:00
&stb_unicode_font_renderer,
#else
&stb_font_renderer,
2016-11-05 15:20:18 +00:00
#endif
#endif
&bitmap_font_renderer,
NULL
};
static void *g_osd_font;
int font_renderer_create_default(const void **data, void **handle,
const char *font_path, unsigned font_size)
{
unsigned i;
2016-02-06 20:51:37 +00:00
const font_renderer_driver_t **drv =
(const font_renderer_driver_t**)data;
for (i = 0; font_backends[i]; i++)
{
const char *path = font_path;
if (!path)
path = font_backends[i]->get_default_font();
if (!path)
continue;
*handle = font_backends[i]->init(path, font_size);
if (*handle)
{
RARCH_LOG("Using font rendering backend: %s.\n",
font_backends[i]->ident);
*drv = font_backends[i];
return 1;
}
else
RARCH_ERR("Failed to create rendering backend: %s.\n",
font_backends[i]->ident);
}
*drv = NULL;
*handle = NULL;
return 0;
}
2012-02-12 14:23:35 +00:00
#ifdef HAVE_D3D
static const font_renderer_t *d3d_font_backends[] = {
2012-12-15 02:35:04 +00:00
#if defined(_XBOX1)
&d3d_xdk1_font,
2012-12-15 03:36:19 +00:00
#elif defined(_XBOX360)
2012-12-15 02:35:04 +00:00
&d3d_xbox360_font,
#elif defined(_WIN32)
&d3d_win32_font,
2012-12-15 02:35:04 +00:00
#endif
};
static bool d3d_font_init_first(
const void **font_driver, void **font_handle,
2015-03-29 22:49:18 +00:00
void *video_data, const char *font_path, float font_size)
2012-12-15 02:35:04 +00:00
{
unsigned i;
for (i = 0; i < ARRAY_SIZE(d3d_font_backends); i++)
2012-12-15 02:35:04 +00:00
{
2016-02-06 20:51:37 +00:00
void *data = d3d_font_backends[i]->init(
video_data, font_path, font_size);
if (!data)
continue;
*font_driver = d3d_font_backends[i];
*font_handle = data;
return true;
2012-12-15 02:35:04 +00:00
}
return false;
2012-12-15 02:35:04 +00:00
}
#endif
#ifdef HAVE_OPENGL
static const font_renderer_t *gl_font_backends[] = {
2015-04-23 00:04:32 +00:00
&gl_raster_font,
#if defined(HAVE_LIBDBGFONT)
&libdbg_font,
#endif
NULL,
};
static bool gl_font_init_first(
const void **font_driver, void **font_handle,
void *video_data, const char *font_path, float font_size)
{
unsigned i;
for (i = 0; gl_font_backends[i]; i++)
{
2016-02-06 20:51:37 +00:00
void *data = gl_font_backends[i]->init(
video_data, font_path, font_size);
if (!data)
continue;
*font_driver = gl_font_backends[i];
*font_handle = data;
return true;
}
return false;
}
#endif
2016-02-16 19:24:00 +00:00
#ifdef HAVE_VULKAN
static const font_renderer_t *vulkan_font_backends[] = {
&vulkan_raster_font,
NULL,
};
static bool vulkan_font_init_first(
const void **font_driver, void **font_handle,
void *video_data, const char *font_path, float font_size)
{
unsigned i;
for (i = 0; vulkan_font_backends[i]; i++)
{
void *data = vulkan_font_backends[i]->init(video_data, font_path, font_size);
if (!data)
continue;
*font_driver = vulkan_font_backends[i];
*font_handle = data;
return true;
}
return false;
}
#endif
2015-10-01 22:11:54 +00:00
#ifdef HAVE_VITA2D
static const font_renderer_t *vita2d_font_backends[] = {
&vita2d_vita_font
};
static bool vita2d_font_init_first(
const void **font_driver, void **font_handle,
void *video_data, const char *font_path, float font_size)
{
unsigned i;
for (i = 0; vita2d_font_backends[i]; i++)
{
2016-02-06 20:51:37 +00:00
void *data = vita2d_font_backends[i]->init(
video_data, font_path, font_size);
2015-10-01 22:11:54 +00:00
if (!data)
continue;
*font_driver = vita2d_font_backends[i];
*font_handle = data;
return true;
}
return false;
}
#endif
#ifdef _3DS
static const font_renderer_t *ctr_font_backends[] = {
&ctr_font
};
static bool ctr_font_init_first(
const void **font_driver, void **font_handle,
void *video_data, const char *font_path, float font_size)
{
unsigned i;
for (i = 0; ctr_font_backends[i]; i++)
{
void *data = ctr_font_backends[i]->init(
video_data, font_path, font_size);
if (!data)
continue;
*font_driver = ctr_font_backends[i];
*font_handle = data;
return true;
}
return false;
}
#endif
2016-02-06 20:51:37 +00:00
static bool font_init_first(
const void **font_driver, void **font_handle,
void *video_data, const char *font_path, float font_size,
enum font_driver_render_api api)
{
if (font_path && !font_path[0])
font_path = NULL;
switch (api)
{
#ifdef HAVE_D3D
case FONT_DRIVER_RENDER_DIRECT3D_API:
return d3d_font_init_first(font_driver, font_handle,
video_data, font_path, font_size);
#endif
#ifdef HAVE_OPENGL
case FONT_DRIVER_RENDER_OPENGL_API:
return gl_font_init_first(font_driver, font_handle,
video_data, font_path, font_size);
2015-10-01 22:11:54 +00:00
#endif
2016-02-16 19:24:00 +00:00
#ifdef HAVE_VULKAN
case FONT_DRIVER_RENDER_VULKAN_API:
return vulkan_font_init_first(font_driver, font_handle,
video_data, font_path, font_size);
#endif
2015-10-01 22:11:54 +00:00
#ifdef HAVE_VITA2D
case FONT_DRIVER_RENDER_VITA2D:
return vita2d_font_init_first(font_driver, font_handle,
video_data, font_path, font_size);
#endif
#ifdef _3DS
case FONT_DRIVER_RENDER_CTR:
return ctr_font_init_first(font_driver, font_handle,
video_data, font_path, font_size);
#endif
case FONT_DRIVER_RENDER_DONT_CARE:
/* TODO/FIXME - lookup graphics driver's 'API' */
break;
default:
break;
}
return false;
}
bool font_driver_has_render_msg(void)
{
return true;
}
2016-02-06 20:51:37 +00:00
void font_driver_render_msg(void *font_data,
const char *msg, const struct font_params *params)
{
font_data_t *font = (font_data_t*)(font_data ? font_data : g_osd_font);
2016-10-19 22:30:34 +00:00
if (font && font->renderer && font->renderer->render_msg)
font->renderer->render_msg(font->renderer_data, msg, params);
}
2015-12-05 10:34:56 +00:00
void font_driver_bind_block(void *font_data, void *block)
2015-12-05 09:59:03 +00:00
{
font_data_t *font = (font_data_t*)(font_data ? font_data : g_osd_font);
2015-12-05 09:59:03 +00:00
2016-10-19 22:30:34 +00:00
if (font && font->renderer && font->renderer->bind_block)
font->renderer->bind_block(font->renderer_data, block);
2015-12-05 09:59:03 +00:00
}
void font_driver_flush(void *font_data)
2015-12-05 11:10:12 +00:00
{
font_data_t *font = (font_data_t*)(font_data ? font_data : g_osd_font);
2016-10-19 22:30:34 +00:00
if (font && font->renderer && font->renderer->flush)
font->renderer->flush(font->renderer_data);
2015-12-05 11:10:12 +00:00
}
int font_driver_get_message_width(void *font_data,
2016-02-06 20:51:37 +00:00
const char *msg, unsigned len, float scale)
2015-12-05 11:17:58 +00:00
{
font_data_t *font = (font_data_t*)(font_data ? font_data : g_osd_font);
2016-10-19 22:30:34 +00:00
if (font && font->renderer && font->renderer->get_message_width)
return font->renderer->get_message_width(font->renderer_data, msg, len, scale);
return -1;
2015-12-05 11:17:58 +00:00
}
void font_driver_free(void *font_data)
{
font_data_t *font = (font_data_t*)font_data;
if (font)
{
if (font->renderer && font->renderer->free)
font->renderer->free(font->renderer_data);
2015-12-05 09:54:53 +00:00
font->renderer = NULL;
font->renderer_data = NULL;
2015-12-05 09:54:53 +00:00
free(font);
}
}
font_data_t *font_driver_init_first(
void *video_data, const char *font_path, float font_size,
bool threading_hint, enum font_driver_render_api api)
{
const void *font_driver;
void *font_handle;
bool ok = false;
2015-12-06 21:42:22 +00:00
#ifdef HAVE_THREADS
settings_t *settings = config_get_ptr();
2015-12-06 21:42:22 +00:00
2016-02-07 19:32:53 +00:00
if (threading_hint
&& settings->video.threaded
2016-05-08 12:00:51 +00:00
&& !video_driver_is_hw_context())
ok = video_thread_font_init(&font_driver, &font_handle,
video_data, font_path, font_size, api, font_init_first);
else
2015-12-06 21:42:22 +00:00
#endif
ok = font_init_first(&font_driver, &font_handle,
video_data, font_path, font_size, api);
if (ok)
{
font_data_t *font = (font_data_t*)calloc(1, sizeof(*font));
font->renderer = (const font_renderer_t*)font_driver;
font->renderer_data = font_handle;
font->size = font_size;
return font;
}
2015-12-05 10:29:06 +00:00
return NULL;
}
2015-12-05 11:10:12 +00:00
void font_driver_init_osd(void *video_data, bool threading_hint, enum font_driver_render_api api)
{
if (!g_osd_font)
{
settings_t *settings = config_get_ptr();
g_osd_font = font_driver_init_first(video_data,
*settings->path.font ? settings->path.font : NULL,
settings->video.font_size, threading_hint, api);
if (!g_osd_font)
RARCH_ERR("[font]: Failed to initialize OSD font.\n");
}
}
void font_driver_free_osd(void)
{
if (g_osd_font)
font_driver_free(g_osd_font);
g_osd_font = NULL;
}