RetroArch/gfx/lima_gfx.c
2014-06-10 02:44:19 +02:00

962 lines
27 KiB
C

/* RetroArch - A frontend for libretro.
* Copyright (C) 2013-2014 - Tobias Jakobi
* Copyright (C) 2013-2014 - Daniel Mehrwald
*
* 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/>.
*/
#include <stdlib.h>
#include <string.h>
#include <limare.h>
#include <GLES2/gl2.h>
#include "../general.h"
#include "gfx_common.h"
#include "fonts/fonts.h"
/* Rename to LIMA_GFX_DEBUG to enable debugging code. */
#define NO_LIMA_GFX_DEBUG 1
/* Current limare only natively supports a limited amount of formats for texture *
* data. We compensate for this limitation by swizzling the texture data in the *
* pixel shader. */
#define LIMA_TEXEL_FORMAT_BGR_565 0x0e
#define LIMA_TEXEL_FORMAT_RGBA_5551 0x0f
#define LIMA_TEXEL_FORMAT_RGBA_4444 0x10
#define LIMA_TEXEL_FORMAT_RGBA_8888 0x16
/* Limare is currently unable to deallocate individual texture objects and *
* only allows to destroy all objects at once. *
* We only create a maximum of 12 objects, before doing a full "reset", or *
* sooner, under the condition that limare's texture memory runs out. */
static const unsigned num_max_textures = 12;
typedef struct limare_state limare_state_t;
typedef struct limare_texture {
unsigned width;
unsigned height;
int handle;
unsigned format;
bool menu;
} limare_texture_t;
typedef struct vec2f {
float x, y;
} vec2f_t;
typedef struct vec3f {
float x, y, z;
} vec3f_t;
/* Create three shader programs. One is for displaying only the emulator core pixel data. *
* The other two are for displaying the menu, where the pixel data can be provided in *
* two different formats. Current RetroArch only seems to ever use a single format, but *
* this is not set in stone, therefore making two programs necessary. */
typedef struct limare_data {
limare_state_t *state;
int program;
int program_menu_rgba16;
int program_menu_rgba32;
float screen_aspect;
float frame_aspect;
unsigned upload_format;
unsigned upload_bpp; /* bytes per pixel */
vec3f_t *vertices;
vec2f_t *coords;
/* Generic buffer to create contiguous pixel data for limare
* or to use for font blitting. */
void *buffer;
unsigned buffer_size;
limare_texture_t **textures;
unsigned texture_slots;
limare_texture_t *cur_texture;
limare_texture_t *cur_texture_menu;
unsigned font_width;
unsigned font_height;
limare_texture_t *font_texture;
} limare_data_t;
/* Header for simple vertex shader. */
static const char *vshader_src =
"attribute vec4 in_vertex;\n"
"attribute vec2 in_coord;\n"
"\n"
"varying vec2 coord;\n"
"\n"
"void main()\n"
"{\n"
" gl_Position = in_vertex;\n"
" coord = in_coord;\n"
"}\n";
/* Header for simple fragment shader. */
static const char *fshader_header_src =
"precision highp float;\n"
"\n"
"varying vec2 coord;\n"
"\n"
"uniform sampler2D in_texture;\n"
"\n";
/* Main (template) for simple fragment shader. */
static const char *fshader_main_src =
"void main()\n"
"{\n"
" vec3 pixel = texture2D(in_texture, coord)%s;\n"
" gl_FragColor = vec4(pixel, 1.0);\n"
"}\n";
/* Header for menu fragment shader. */
/* Use mediump, which makes uColor into a (single-precision) float[4]. */
static const char *fshader_menu_header_src =
"precision mediump float;\n"
"\n"
"varying vec2 coord;\n"
"uniform vec4 uColor;\n"
"\n"
"uniform sampler2D in_texture;\n"
"\n";
/* Main (template) for menu fragment shader. */
static const char *fshader_menu_main_src =
"void main()\n"
"{\n"
" vec4 pixel = texture2D(in_texture, coord)%s;\n"
" gl_FragColor = pixel * uColor;\n"
"}\n";
static inline void put_pixel_rgba4444(uint16_t *p, unsigned r, unsigned g, unsigned b, unsigned a) {
*p = (a >> 4) | ((b >> 4) << 4) | ((g >> 4) << 8) | ((r >> 4) << 12);
}
static inline unsigned align_common(unsigned i, unsigned j) {
return (i + j - 1) & ~(j - 1);
}
static float get_screen_aspect(limare_state_t *state) {
unsigned w = 0, h = 0;
limare_buffer_size(state, &w, &h);
if (w != 0 && h != 0) {
return (float)w / (float)h;
}
return 0.0f;
}
static void apply_aspect(limare_data_t *pdata, float ratio) {
vec3f_t *vertices = pdata->vertices;
float x, y;
if (fabsf(pdata->screen_aspect - pdata->frame_aspect) < 0.0001f) {
x = 1.0f;
y = 1.0f;
} else {
if (pdata->screen_aspect > pdata->frame_aspect) {
x = pdata->frame_aspect / pdata->screen_aspect;
y = 1.0f;
} else {
x = 1.0f;
y = pdata->screen_aspect / pdata->frame_aspect;
}
}
/* TODO: use ratio parameter */
vertices[0].x = vertices[2].x = -x;
vertices[1].x = vertices[3].x = x;
vertices[0].y = vertices[1].y = -y;
vertices[2].y = vertices[3].y = y;
}
static int destroy_textures(limare_data_t *pdata) {
unsigned i;
int ret;
pdata->cur_texture = NULL;
pdata->cur_texture_menu = NULL;
for (i = 0; i < pdata->texture_slots; ++i) {
free(pdata->textures[i]);
pdata->textures[i] = NULL;
}
ret = limare_texture_cleanup(pdata->state);
pdata->texture_slots = 0;
return ret;
}
static limare_texture_t *get_texture_handle(limare_data_t *pdata,
unsigned width, unsigned height, unsigned format) {
unsigned i;
format = (format == 0) ? pdata->upload_format : format;
for (i = 0; i < pdata->texture_slots; ++i) {
if (pdata->textures[i]->width == width &&
pdata->textures[i]->height == height &&
pdata->textures[i]->format == format) return pdata->textures[i];
}
if (pdata->texture_slots == num_max_textures) {
/* All texture slots are used, do a reset. */
if (destroy_textures(pdata)) {
RARCH_ERR("video_lima: failed to reset texture storage\n");
}
}
return NULL;
}
static limare_texture_t *add_texture(limare_data_t *pdata,
unsigned width, unsigned height,
const void *pixels, unsigned format) {
int texture = -1;
unsigned retries = 2;
const unsigned i = pdata->texture_slots;
format = (format == 0) ? pdata->upload_format : format;
/* limare_texture_upload returns -1 when the upload fails for some reason. */
while (texture == -1 && retries > 0) {
texture = limare_texture_upload(pdata->state, pixels, width, height, format, 0);
if (texture != -1) break;
destroy_textures(pdata);
retries--;
}
if (texture == -1) return NULL;
/* Set magnification to linear and minification to nearest, since we will *
* probably only ever scale the image to larger dimensions. Also set *
* wrap mode for both coords to clamp, which should eliminate some artifacts. */
limare_texture_parameters(pdata->state, texture, GL_LINEAR, GL_NEAREST,
GL_CLAMP_TO_EDGE, GL_CLAMP_TO_EDGE);
pdata->textures[i] = calloc(1, sizeof(limare_texture_t));
pdata->textures[i]->width = width;
pdata->textures[i]->height = height;
pdata->textures[i]->handle = texture;
pdata->textures[i]->format = format;
pdata->texture_slots++;
return pdata->textures[i];
}
static const void *make_contiguous(limare_data_t *pdata,
unsigned width, unsigned height,
const void *pixels, unsigned bpp,
unsigned pitch) {
unsigned i;
unsigned full_pitch;
bpp = (bpp == 0) ? pdata->upload_bpp : bpp;
full_pitch = width * bpp;
if (full_pitch == pitch) return pixels;
RARCH_LOG("video_lima: input buffer not contiguous\n");
/* Enlarge our buffer, if it is currently too small. */
if (pdata->buffer_size < full_pitch * height) {
const unsigned aligned_size = align_common(full_pitch * height, 16);
free(pdata->buffer);
pdata->buffer = NULL;
posix_memalign(&pdata->buffer, 16, aligned_size);
if (pdata->buffer == NULL) {
RARCH_ERR("video_lima: failed to allocate buffer to make pixel data contiguous\n");
return NULL;
}
pdata->buffer_size = aligned_size;
}
for (i = 0; i < height; ++i) {
memcpy(pdata->buffer + i * full_pitch, pixels + i * pitch, full_pitch);
}
return pdata->buffer;
}
#ifdef LIMA_GFX_DEBUG
static void print_status(limare_data_t *pdata) {
unsigned i;
RARCH_LOG("video_lima: upload format = 0x%x, upload bpp = %u\n", pdata->upload_format, pdata->upload_bpp);
RARCH_LOG("video_lima: buffer at %p, buffer size = %u\n", pdata->buffer, pdata->buffer_size);
RARCH_LOG("video_lima: used texture slots = %u (from %u)\n", pdata->texture_slots, num_max_textures);
for (i = 0; i < pdata->texture_slots; ++i) {
RARCH_LOG("video_lima: texture slot %u, width = %u, height = %u, handle = %u, format = 0x%x\n",
i, pdata->textures[i]->width, pdata->textures[i]->height,
pdata->textures[i]->handle, pdata->textures[i]->format);
}
}
#endif
static void destroy_data(limare_data_t *pdata) {
free(pdata->vertices);
free(pdata->coords);
}
static int setup_data(limare_data_t *pdata) {
static const unsigned num_verts = 4;
static const unsigned num_coords = 4 * 4;
unsigned i;
static const vec3f_t vertices[4] = {
{-1.0f, -1.0f, 0.0f},
{ 1.0f, -1.0f, 0.0f},
{-1.0f, 1.0f, 0.0f},
{ 1.0f, 1.0f, 0.0f}
};
static const vec2f_t coords[16] = {
{0.0f, 1.0f}, {1.0f, 1.0f}, /* 0 degrees */
{0.0f, 0.0f}, {1.0f, 0.0f},
{0.0f, 0.0f}, {0.0f, 1.0f}, /* 90 degrees */
{1.0f, 0.0f}, {1.0f, 1.0f},
{1.0f, 0.0f}, {0.0f, 0.0f}, /* 180 degrees */
{1.0f, 1.0f}, {0.0f, 1.0f},
{1.0f, 1.0f}, {1.0f, 0.0f}, /* 270 degrees */
{0.0f, 1.0f}, {0.0f, 0.0f}
};
pdata->vertices = calloc(num_verts, sizeof(vec3f_t));
if (pdata->vertices == NULL) goto fail;
pdata->coords = calloc(num_coords, sizeof(vec2f_t));
if (pdata->coords == NULL) goto fail;
for (i = 0; i < num_verts; ++i) {
pdata->vertices[i] = vertices[i];
}
for (i = 0; i < num_coords; ++i) {
pdata->coords[i] = coords[i];
}
return 0;
fail:
return -1;
}
static int create_programs(limare_data_t *pdata) {
char tmpbufm[1024]; /* temp buffer for main function */
char tmpbuf[1024]; /* temp buffer for whole program */
const char* swz = (pdata->upload_bpp == 4) ? ".bgr" : ".rgb";
/* Create shader program for regular operation first. */
pdata->program = limare_program_new(pdata->state);
if (pdata->program < 0) goto fail;
snprintf(tmpbufm, 1024, fshader_main_src, swz);
strncpy(tmpbuf, fshader_header_src, 1024);
strcat(tmpbuf, tmpbufm);
if (vertex_shader_attach(pdata->state, pdata->program, vshader_src)) goto fail;
if (fragment_shader_attach(pdata->state, pdata->program, tmpbuf)) goto fail;
if (limare_link(pdata->state)) goto fail;
/* Create shader program for menu with RGBA4444 pixel data. */
pdata->program_menu_rgba16 = limare_program_new(pdata->state);
if (pdata->program_menu_rgba16 < 0) goto fail;
snprintf(tmpbufm, 1024, fshader_menu_main_src, ".abgr");
strncpy(tmpbuf, fshader_menu_header_src, 1024);
strcat(tmpbuf, tmpbufm);
if (vertex_shader_attach(pdata->state, pdata->program_menu_rgba16, vshader_src)) goto fail;
if (fragment_shader_attach(pdata->state, pdata->program_menu_rgba16, tmpbuf)) goto fail;
if (limare_link(pdata->state)) goto fail;
/* Create shader program for menu with RGBA8888 pixel data. */
pdata->program_menu_rgba32 = limare_program_new(pdata->state);
if (pdata->program_menu_rgba32 < 0) goto fail;
snprintf(tmpbufm, 1024, fshader_menu_main_src, ".abgr");
strncpy(tmpbuf, fshader_menu_header_src, 1024);
strcat(tmpbuf, tmpbufm);
if (vertex_shader_attach(pdata->state, pdata->program_menu_rgba32, vshader_src)) goto fail;
if (fragment_shader_attach(pdata->state, pdata->program_menu_rgba32, tmpbuf)) goto fail;
if (limare_link(pdata->state)) goto fail;
return 0;
fail:
return -1;
}
static void put_glyph_rgba4444(limare_data_t *pdata, const uint8_t *src, uint8_t *f_rgb,
unsigned g_width, unsigned g_height, unsigned g_pitch,
unsigned dst_x, unsigned dst_y) {
unsigned x, y;
uint16_t *dst;
dst = (uint16_t*)pdata->buffer + dst_y * pdata->font_width + dst_x;
for (y = 0; y < g_height; ++y, src += g_pitch, dst += pdata->font_width) {
for (x = 0; x < g_width; ++x) {
const uint8_t blend = src[x];
if (blend != 0) put_pixel_rgba4444(&dst[x], f_rgb[0], f_rgb[1], f_rgb[2], blend);
}
}
}
typedef struct lima_video {
limare_data_t *lima;
void *font;
const font_renderer_driver_t *font_driver;
uint8_t font_rgb[4];
/* current dimensions */
unsigned width;
unsigned height;
/* MENU data */
int menu_rotation;
float menu_alpha;
bool menu_active;
bool menu_rgb32;
bool aspect_changed;
} lima_video_t;
static void lima_gfx_free(void *data) {
lima_video_t *vid = data;
if (!vid) return;
if (vid->lima && vid->lima->state) limare_finish(vid->lima->state);
if (vid->font) vid->font_driver->free(vid->font);
destroy_data(vid->lima);
destroy_textures(vid->lima);
free(vid->lima->textures);
free(vid->lima);
free(vid);
}
static void lima_init_font(lima_video_t *vid, const char *font_path, unsigned font_size) {
if (!g_settings.video.font_enable) return;
if (font_renderer_create_default(&vid->font_driver, &vid->font,
*g_settings.video.font_path ? g_settings.video.font_path : NULL, g_settings.video.font_size)) {
int r = g_settings.video.msg_color_r * 255;
int g = g_settings.video.msg_color_g * 255;
int b = g_settings.video.msg_color_b * 255;
vid->font_rgb[0] = r < 0 ? 0 : (r > 255 ? 255 : r);
vid->font_rgb[1] = g < 0 ? 0 : (g > 255 ? 255 : g);
vid->font_rgb[2] = b < 0 ? 0 : (b > 255 ? 255 : b);
} else {
RARCH_LOG("video_lima: font init failed\n");
}
}
static void lima_render_msg(lima_video_t *vid, const char *msg) {
struct font_output_list out;
struct font_output *head;
unsigned req_size;
limare_data_t *lima = vid->lima;
int msg_base_x = g_settings.video.msg_pos_x * lima->font_width;
int msg_base_y = (1.0 - g_settings.video.msg_pos_y) * lima->font_height;
if (vid->font == NULL) return;
/* Font texture uses RGBA4444 pixel data (2 bytes per pixel). */
req_size = lima->font_width * lima->font_height * 2;
if (lima->buffer_size < req_size) {
const unsigned aligned_size = align_common(req_size, 16);
free(lima->buffer);
lima->buffer = NULL;
posix_memalign(&lima->buffer, 16, aligned_size);
if (lima->buffer == NULL) {
RARCH_ERR("video_lima: failed to allocate buffer to render fonts\n");
return;
}
lima->buffer_size = aligned_size;
}
memset(lima->buffer, 0, req_size);
/* FIXME: Untested new font rendering code. */
const struct font_atlas *atlas = vid->font_driver->get_atlas(vid->font);
for (; msg; msg++) {
const struct font_glyph *glyph = vid->font_driver->get_glyph(vid->font, (uint8_t)*msg);
if (!glyph)
continue;
int base_x = msg_base_x + glyph->draw_offset_x;
int base_y = msg_base_y + glyph->draw_offset_y;
const int max_width = lima->font_width - base_x;
const int max_height = lima->font_height - base_y;
int glyph_width = head->width;
int glyph_height = head->height;
const uint8_t *src = atlas->buffer + glyph->atlas_offset_x + glyph->atlas_offset_y * atlas->width;
if (base_x < 0) {
src -= base_x;
glyph_width += base_x;
base_x = 0;
}
if (base_y < 0) {
src -= base_y * (int)atlas->width;
glyph_height += base_y;
base_y = 0;
}
if (max_width <= 0 || max_height <= 0) continue;
if (glyph_width > max_width) glyph_width = max_width;
if (glyph_height > max_height) glyph_height = max_height;
put_glyph_rgba4444(lima, src, vid->font_rgb,
glyph_width, glyph_height,
atlas->width, base_x, base_y);
msg_base_x += glyph->advance_x;
msg_base_y += glyph->advance_y;
}
}
static void *lima_gfx_init(const video_info_t *video, const input_driver_t **input, void **input_data) {
lima_video_t *vid = NULL;
limare_data_t *lima = NULL;
void *lima_input = NULL;
struct limare_windowsys_drm limare_config = { 0 };
vid = calloc(1, sizeof(lima_video_t));
if (!vid) return NULL;
vid->menu_alpha = 1.0f;
lima = calloc(1, sizeof(limare_data_t));
if (!lima) goto fail;
/* Request the Exynos DRM backend for rendering. */
limare_config.type = LIMARE_WINDOWSYS_DRM;
limare_config.connector_index = g_settings.video.monitor_index;
lima->state = limare_init(&limare_config);
if (!lima->state) {
RARCH_ERR("video_lima: limare initialization failed\n");
goto fail;
}
limare_buffer_clear(lima->state);
if (limare_state_setup(lima->state, g_settings.video.fullscreen_x,
g_settings.video.fullscreen_y, 0xff000000)) {
RARCH_ERR("video_lima: limare state setup failed\n");
goto fail_lima;
}
lima->screen_aspect = get_screen_aspect(lima->state);
lima->font_height = 368;
lima->font_width = align_common((unsigned)(lima->screen_aspect * (float)lima->font_height), 16);
lima->upload_format = video->rgb32 ?
LIMA_TEXEL_FORMAT_RGBA_8888 : LIMA_TEXEL_FORMAT_BGR_565;
lima->upload_bpp = video->rgb32 ? 4 : 2;
limare_enable(lima->state, GL_DEPTH_TEST);
limare_depth_func(lima->state, GL_ALWAYS);
limare_depth_mask(lima->state, GL_TRUE);
limare_enable(lima->state, GL_CULL_FACE);
limare_blend_func(lima->state, GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
if (setup_data(lima)) {
RARCH_ERR("video_lima: data setup failed\n");
goto fail_lima;
}
if (create_programs(lima)) {
RARCH_ERR("video_lima: creating shader programs failed\n");
goto fail_lima;
}
lima->textures = calloc(num_max_textures, sizeof(limare_texture_t*));
if (input && input_data) {
*input = NULL;
*input_data = NULL;
}
vid->lima = lima;
lima_init_font(vid, g_settings.video.font_path, g_settings.video.font_size);
return vid;
fail_lima:
limare_finish(lima->state);
fail:
free(lima);
free(vid);
return NULL;
}
static bool lima_gfx_frame(void *data, const void *frame,
unsigned width, unsigned height,
unsigned pitch, const char *msg) {
lima_video_t *vid;
const void *pixels;
limare_data_t *lima;
bool upload_frame = true;
vid = data;
/* Check if neither menu nor emulator framebuffer is to be displayed. */
if (!vid->menu_active && frame == NULL) return true;
lima = vid->lima;
if (frame != NULL) {
/* Handle resolution changes from the emulation core. */
if (width != vid->width || height != vid->height) {
limare_texture_t *tex;
if (width == 0 || height == 0) return true;
RARCH_LOG("video_lima: resolution was changed by core to %ux%u\n", width, height);
tex = get_texture_handle(lima, width, height, 0);
if (tex == NULL) {
pixels = make_contiguous(lima, width, height, frame, 0, pitch);
tex = add_texture(lima, width, height, pixels, 0);
if (tex == NULL) {
RARCH_ERR("video_lima: failed to allocate new texture with dimensions %ux%u\n",
width, height);
return false;
}
upload_frame = false; /* pixel data already got uploaded during texture allocation */
}
lima->cur_texture = tex;
vid->width = width;
vid->height = height;
lima->frame_aspect = (float)width / (float)height;
vid->aspect_changed = true;
}
if (upload_frame) {
pixels = make_contiguous(lima, width, height, frame, 0, pitch);
limare_texture_mipmap_upload(lima->state, lima->cur_texture->handle, 0, pixels);
}
}
if (g_settings.fps_show) {
char buffer[128], buffer_fps[128];
gfx_get_fps(buffer, sizeof(buffer), g_settings.fps_show ? buffer_fps : NULL, sizeof(buffer_fps));
msg_queue_push(g_extern.msg_queue, buffer_fps, 1, 1);
}
if (vid->aspect_changed) {
apply_aspect(lima, g_extern.system.aspect_ratio);
vid->aspect_changed = false;
}
limare_frame_new(lima->state);
if (lima->cur_texture != NULL) {
limare_program_current(lima->state, lima->program);
limare_attribute_pointer(lima->state, "in_vertex", LIMARE_ATTRIB_FLOAT,
3, 0, 4, lima->vertices);
limare_attribute_pointer(lima->state, "in_coord", LIMARE_ATTRIB_FLOAT,
2, 0, 4, lima->coords + vid->menu_rotation * 4);
limare_texture_attach(lima->state, "in_texture", lima->cur_texture->handle);
if (limare_draw_arrays(lima->state, GL_TRIANGLE_STRIP, 0, 4)) return false;
}
/* Handle font rendering. */
if (msg) {
bool upload_font = true;
/* Both font_vertices and font_color are constant, but we can't make them *
* const, since limare_attribute_pointer expects (non-const) void pointers. */
static vec3f_t font_vertices[4] = {
{-1.0f, -1.0f, 0.0f},
{ 1.0f, -1.0f, 0.0f},
{-1.0f, 1.0f, 0.0f},
{ 1.0f, 1.0f, 0.0f}
};
static float font_color[4] = {1.0f, 1.0f, 1.0f, 1.0f};
lima_render_msg(vid, msg);
if (lima->font_texture == NULL) {
lima->font_texture = add_texture(lima, lima->font_width, lima->font_height,
lima->buffer, LIMA_TEXEL_FORMAT_RGBA_4444);
upload_font = false;
}
if (upload_font)
limare_texture_mipmap_upload(lima->state, lima->font_texture->handle, 0, lima->buffer);
/* We re-use the RGBA16 menu program here. */
limare_program_current(lima->state, lima->program_menu_rgba16);
limare_attribute_pointer(lima->state, "in_vertex", LIMARE_ATTRIB_FLOAT,
3, 0, 4, font_vertices);
limare_attribute_pointer(lima->state, "in_coord", LIMARE_ATTRIB_FLOAT,
2, 0, 4, lima->coords + vid->menu_rotation * 4);
limare_texture_attach(lima->state, "in_texture", lima->font_texture->handle);
limare_uniform_attach(lima->state, "uColor", 4, font_color);
limare_enable(lima->state, GL_BLEND);
if (limare_draw_arrays(lima->state, GL_TRIANGLE_STRIP, 0, 4)) return false;
limare_disable(lima->state, GL_BLEND);
}
if (vid->menu_active && lima->cur_texture_menu != NULL) {
float color[4] = {1.0f, 1.0f, 1.0f, vid->menu_alpha};
if (vid->menu_rgb32)
limare_program_current(lima->state, lima->program_menu_rgba32);
else
limare_program_current(lima->state, lima->program_menu_rgba16);
limare_attribute_pointer(lima->state, "in_vertex", LIMARE_ATTRIB_FLOAT,
3, 0, 4, lima->vertices);
limare_attribute_pointer(lima->state, "in_coord", LIMARE_ATTRIB_FLOAT,
2, 0, 4, lima->coords + vid->menu_rotation * 4);
limare_texture_attach(lima->state, "in_texture", lima->cur_texture_menu->handle);
limare_uniform_attach(lima->state, "uColor", 4, color);
limare_enable(lima->state, GL_BLEND);
if (limare_draw_arrays(lima->state, GL_TRIANGLE_STRIP, 0, 4)) return false;
limare_disable(lima->state, GL_BLEND);
}
if (limare_frame_flush(lima->state)) return false;
limare_buffer_swap(lima->state);
g_extern.frame_count++;
#ifdef LIMA_GFX_DEBUG
print_status(lima);
#endif
return true;
}
static void lima_gfx_set_nonblock_state(void *data, bool state) {
(void)data; /* limare doesn't export vsync control yet */
(void)state;
}
static bool lima_gfx_alive(void *data) {
(void)data;
return true; /* always alive */
}
static bool lima_gfx_focus(void *data) {
(void)data;
return true; /* limare doesn't use windowing, so we always have focus */
}
static void lima_gfx_set_rotation(void *data, unsigned rotation) {
lima_video_t *vid = data;
vid->menu_rotation = rotation;
}
static void lima_gfx_viewport_info(void *data, struct rarch_viewport *vp) {
lima_video_t *vid = data;
vp->x = vp->y = 0;
vp->width = vp->full_width = vid->width;
vp->height = vp->full_height = vid->height;
}
static void lima_set_aspect_ratio(void *data, unsigned aspect_ratio_idx) {
lima_video_t *vid = data;
switch (aspect_ratio_idx) {
case ASPECT_RATIO_SQUARE:
gfx_set_square_pixel_viewport(g_extern.system.av_info.geometry.base_width, g_extern.system.av_info.geometry.base_height);
break;
case ASPECT_RATIO_CORE:
gfx_set_core_viewport();
break;
case ASPECT_RATIO_CONFIG:
gfx_set_config_viewport();
break;
default:
break;
}
g_extern.system.aspect_ratio = aspectratio_lut[aspect_ratio_idx].value;
vid->aspect_changed = true;
}
static void lima_apply_state_changes(void *data) {
(void)data;
}
static void lima_set_texture_frame(void *data, const void *frame, bool rgb32,
unsigned width, unsigned height, float alpha) {
lima_video_t *vid = data;
limare_texture_t* tex;
const unsigned format = rgb32 ? LIMA_TEXEL_FORMAT_RGBA_8888 :
LIMA_TEXEL_FORMAT_RGBA_4444;
vid->menu_rgb32 = rgb32;
vid->menu_alpha = alpha;
tex = vid->lima->cur_texture_menu;
/* Current menu doesn't change dimensions, so we should hit this most of the time. */
if (tex != NULL && tex->width == width &&
tex->height == height && tex->format == format) goto upload;
if (tex == NULL) {
tex = get_texture_handle(vid->lima, width, height, format);
if (tex == NULL) {
tex = add_texture(vid->lima, width, height, frame, format);
if (tex != NULL) {
vid->lima->cur_texture_menu = tex;
goto upload;
}
RARCH_ERR("video_lima: failed to allocate new menu texture with dimensions %ux%u\n",
width, height);
}
}
return;
upload:
limare_texture_mipmap_upload(vid->lima->state, tex->handle, 0, frame);
}
static void lima_set_texture_enable(void *data, bool state, bool full_screen) {
lima_video_t *vid = data;
vid->menu_active = state;
}
static void lima_set_osd_msg(void *data, const char *msg, void *userdata) {
lima_video_t *vid = data;
/* TODO: what does this do? */
(void)msg;
(void)userdata;
}
static void lima_show_mouse(void *data, bool state) {
(void)data;
}
static const video_poke_interface_t lima_poke_interface = {
NULL, /* set_filtering */
#ifdef HAVE_FBO
NULL, /* get_current_framebuffer */
NULL, /* get_proc_address */
#endif
lima_set_aspect_ratio,
lima_apply_state_changes,
#ifdef HAVE_MENU
lima_set_texture_frame,
lima_set_texture_enable,
#endif
lima_set_osd_msg,
lima_show_mouse
};
static void lima_gfx_get_poke_interface(void *data, const video_poke_interface_t **iface) {
(void)data;
*iface = &lima_poke_interface;
}
const video_driver_t video_lima = {
lima_gfx_init,
lima_gfx_frame,
lima_gfx_set_nonblock_state,
lima_gfx_alive,
lima_gfx_focus,
NULL, /* set_shader */
lima_gfx_free,
"lima",
lima_gfx_set_rotation,
lima_gfx_viewport_info,
NULL, /* read_viewport */
#ifdef HAVE_OVERLAY
NULL, /* overlay_interface */
#endif
lima_gfx_get_poke_interface
};