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