2015-03-13 04:59:52 +00:00
|
|
|
/* RetroArch - A frontend for libretro.
|
|
|
|
* Copyright (C) 2015 - Manuel Alfayate
|
|
|
|
*
|
|
|
|
* 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 <bcm_host.h>
|
2015-11-22 12:32:46 +00:00
|
|
|
|
2015-03-15 04:23:13 +00:00
|
|
|
#include <rthreads/rthreads.h>
|
|
|
|
|
2015-03-13 04:59:52 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
2016-09-11 12:46:53 +00:00
|
|
|
#include "../../config.h"
|
2015-03-13 04:59:52 +00:00
|
|
|
#endif
|
|
|
|
|
2016-09-05 16:31:32 +00:00
|
|
|
#include "../../configuration.h"
|
2015-11-22 12:32:46 +00:00
|
|
|
#include "../../driver.h"
|
|
|
|
#include "../../retroarch.h"
|
2016-09-11 12:46:53 +00:00
|
|
|
#include "../../runloop.h"
|
2015-11-22 12:32:46 +00:00
|
|
|
#include "../video_context_driver.h"
|
2015-12-05 11:07:22 +00:00
|
|
|
#include "../font_driver.h"
|
2015-11-22 12:32:46 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
struct dispmanx_page
|
2015-05-09 13:41:10 +00:00
|
|
|
{
|
2015-06-16 19:30:16 +00:00
|
|
|
/* Each page contains it's own resource handler
|
|
|
|
* instead of pointing to in by page number */
|
|
|
|
DISPMANX_RESOURCE_HANDLE_T resource;
|
|
|
|
bool used;
|
|
|
|
/* Each page has it's own mutex for
|
|
|
|
* isolating it's used flag access. */
|
|
|
|
slock_t *page_used_mutex;
|
|
|
|
|
|
|
|
/* This field will allow us to access the
|
|
|
|
* main _dispvars struct from the vsync CB function */
|
|
|
|
struct dispmanx_video *dispvars;
|
2015-06-20 14:56:25 +00:00
|
|
|
|
|
|
|
/* This field will allow us to access the
|
|
|
|
* surface the page belongs to. */
|
|
|
|
struct dispmanx_surface *surface;
|
2015-06-16 19:30:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct dispmanx_surface
|
|
|
|
{
|
|
|
|
/* main surface has 3 pages, menu surface has 1 */
|
|
|
|
unsigned int numpages;
|
|
|
|
struct dispmanx_page *pages;
|
2015-06-20 14:56:25 +00:00
|
|
|
/* the page that's currently on screen */
|
|
|
|
struct dispmanx_page *current_page;
|
2015-06-16 19:30:16 +00:00
|
|
|
unsigned int bpp;
|
|
|
|
|
|
|
|
VC_RECT_T src_rect;
|
|
|
|
VC_RECT_T dst_rect;
|
|
|
|
VC_RECT_T bmp_rect;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
/* Each surface has it's own element, and the
|
|
|
|
* resources are contained one in each page */
|
2015-03-13 21:39:18 +00:00
|
|
|
DISPMANX_ELEMENT_HANDLE_T element;
|
2015-06-16 19:30:16 +00:00
|
|
|
VC_DISPMANX_ALPHA_T alpha;
|
|
|
|
VC_IMAGE_TYPE_T pixformat;
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
/* Surfaces with a higher layer will be on top of
|
|
|
|
* the ones with lower. Default is 0. */
|
2015-06-16 19:30:16 +00:00
|
|
|
int layer;
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
/* We need to keep this value for the blitting on
|
|
|
|
* the surface_update function. */
|
2015-06-16 19:30:16 +00:00
|
|
|
int pitch;
|
|
|
|
};
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
struct dispmanx_video
|
|
|
|
{
|
|
|
|
DISPMANX_DISPLAY_HANDLE_T display;
|
|
|
|
DISPMANX_UPDATE_HANDLE_T update;
|
|
|
|
uint32_t vc_image_ptr;
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
struct dispmanx_surface *main_surface;
|
|
|
|
struct dispmanx_surface *menu_surface;
|
|
|
|
struct dispmanx_surface *back_surface;
|
2015-03-13 21:39:18 +00:00
|
|
|
|
|
|
|
/* For console blanking */
|
2015-06-16 19:30:16 +00:00
|
|
|
int fb_fd;
|
|
|
|
uint8_t *fb_addr;
|
2015-03-13 21:39:18 +00:00
|
|
|
unsigned int screensize;
|
2015-06-16 19:30:16 +00:00
|
|
|
uint8_t *screen_bck;
|
|
|
|
|
|
|
|
/* Total dispmanx video dimensions. Not counting overscan settings. */
|
|
|
|
unsigned int dispmanx_width;
|
|
|
|
unsigned int dispmanx_height;
|
2015-03-13 21:39:18 +00:00
|
|
|
|
|
|
|
/* For threading */
|
2015-03-15 04:23:13 +00:00
|
|
|
scond_t *vsync_condition;
|
2015-06-16 19:30:16 +00:00
|
|
|
slock_t *vsync_cond_mutex;
|
2015-03-15 04:23:13 +00:00
|
|
|
slock_t *pending_mutex;
|
2015-06-16 19:30:16 +00:00
|
|
|
unsigned int pageflip_pending;
|
2015-03-13 21:39:18 +00:00
|
|
|
|
|
|
|
/* Menu */
|
|
|
|
bool menu_active;
|
2015-06-27 19:14:30 +00:00
|
|
|
|
|
|
|
bool rgb32;
|
|
|
|
|
|
|
|
/* We use this to keep track of internal resolution changes
|
|
|
|
* done by cores in the main surface or in the menu.
|
|
|
|
* We need these outside the surface because we free surfaces
|
|
|
|
* and then we want to test if these values have changed before
|
|
|
|
* recreating them. */
|
|
|
|
int core_width;
|
|
|
|
int core_height;
|
|
|
|
int core_pitch;
|
|
|
|
int menu_width;
|
|
|
|
int menu_height;
|
|
|
|
int menu_pitch;
|
|
|
|
/* Both main and menu surfaces are going to have the same aspect,
|
|
|
|
* so we keep it here for future reference. */
|
2016-06-15 12:28:13 +00:00
|
|
|
float aspect_ratio;
|
2015-03-13 04:59:52 +00:00
|
|
|
};
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* If no free page is available when called, wait for a page flip. */
|
2015-11-28 16:00:44 +00:00
|
|
|
static struct dispmanx_page *dispmanx_get_free_page(void *data, struct dispmanx_surface *surface)
|
|
|
|
{
|
2015-06-16 19:30:16 +00:00
|
|
|
unsigned i;
|
2015-03-13 21:39:18 +00:00
|
|
|
struct dispmanx_video *_dispvars = data;
|
2015-06-16 19:30:16 +00:00
|
|
|
struct dispmanx_page *page = NULL;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
while (!page)
|
|
|
|
{
|
|
|
|
/* Try to find a free page */
|
2015-11-28 16:00:44 +00:00
|
|
|
for (i = 0; i < surface->numpages; ++i)
|
|
|
|
{
|
2015-06-16 19:30:16 +00:00
|
|
|
if (!surface->pages[i].used)
|
|
|
|
{
|
|
|
|
page = (surface->pages) + i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* If no page is free at the moment,
|
|
|
|
* wait until a free page is freed by vsync CB. */
|
2015-11-28 16:00:44 +00:00
|
|
|
if (!page)
|
|
|
|
{
|
|
|
|
slock_lock(_dispvars->vsync_cond_mutex);
|
|
|
|
scond_wait(_dispvars->vsync_condition, _dispvars->vsync_cond_mutex);
|
|
|
|
slock_unlock(_dispvars->vsync_cond_mutex);
|
2015-06-16 19:30:16 +00:00
|
|
|
}
|
|
|
|
}
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* We mark the choosen page as used */
|
|
|
|
slock_lock(page->page_used_mutex);
|
|
|
|
page->used = true;
|
|
|
|
slock_unlock(page->page_used_mutex);
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
return page;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
static void dispmanx_vsync_callback(DISPMANX_UPDATE_HANDLE_T u, void *data)
|
2015-03-13 04:59:52 +00:00
|
|
|
{
|
2015-06-16 19:30:16 +00:00
|
|
|
struct dispmanx_page *page = data;
|
2015-06-20 14:56:25 +00:00
|
|
|
struct dispmanx_surface *surface = page->surface;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* Marking the page as free must be done before the signaling
|
|
|
|
* so when update_main continues (it won't continue until we signal)
|
|
|
|
* we can chose this page as free */
|
2015-11-28 16:00:44 +00:00
|
|
|
if (surface->current_page)
|
|
|
|
{
|
2015-06-20 14:56:25 +00:00
|
|
|
slock_lock(surface->current_page->page_used_mutex);
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* We mark as free the page that was visible until now */
|
2015-06-20 14:56:25 +00:00
|
|
|
surface->current_page->used = false;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-06-20 14:56:25 +00:00
|
|
|
slock_unlock(surface->current_page->page_used_mutex);
|
2015-06-16 19:30:16 +00:00
|
|
|
}
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
/* The page on which we issued the flip that
|
|
|
|
* caused this callback becomes the visible one */
|
2015-06-20 14:56:25 +00:00
|
|
|
surface->current_page = page;
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
/* These two things must be isolated "atomically" to avoid getting
|
|
|
|
* a false positive in the pending_mutex test in update_main. */
|
2015-06-16 19:30:16 +00:00
|
|
|
slock_lock(page->dispvars->pending_mutex);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
page->dispvars->pageflip_pending--;
|
|
|
|
scond_signal(page->dispvars->vsync_condition);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
slock_unlock(page->dispvars->pending_mutex);
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
static void dispmanx_surface_free(void *data, struct dispmanx_surface **sp)
|
2015-03-13 21:39:18 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct dispmanx_video *_dispvars = data;
|
2015-06-27 19:14:30 +00:00
|
|
|
struct dispmanx_surface *surface = *sp;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
/* What if we run into the vsync cb code after freeing the surface?
|
|
|
|
* We could be trying to get non-existant lock, signal non-existant condition..
|
|
|
|
* So we wait for any pending flips to complete before freeing any surface. */
|
|
|
|
slock_lock(_dispvars->pending_mutex);
|
|
|
|
if (_dispvars->pageflip_pending > 0)
|
|
|
|
scond_wait(_dispvars->vsync_condition, _dispvars->pending_mutex);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
slock_unlock(_dispvars->pending_mutex);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
|
|
|
for (i = 0; i < surface->numpages; i++)
|
|
|
|
{
|
2015-06-18 12:31:42 +00:00
|
|
|
vc_dispmanx_resource_delete(surface->pages[i].resource);
|
|
|
|
surface->pages[i].used = false;
|
|
|
|
slock_free(surface->pages[i].page_used_mutex);
|
2015-06-16 19:30:16 +00:00
|
|
|
}
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-18 12:31:42 +00:00
|
|
|
free(surface->pages);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-18 12:31:42 +00:00
|
|
|
_dispvars->update = vc_dispmanx_update_start(0);
|
2015-06-16 19:30:16 +00:00
|
|
|
vc_dispmanx_element_remove(_dispvars->update, surface->element);
|
2015-03-13 21:39:18 +00:00
|
|
|
vc_dispmanx_update_submit_sync(_dispvars->update);
|
2015-06-18 12:31:42 +00:00
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
free(surface);
|
|
|
|
*sp = NULL;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
static void dispmanx_surface_setup(void *data, int src_width, int src_height,
|
|
|
|
int visible_pitch, int bpp, VC_IMAGE_TYPE_T pixformat,
|
|
|
|
int alpha, float aspect, int numpages, int layer,
|
|
|
|
struct dispmanx_surface **sp)
|
2015-03-13 21:39:18 +00:00
|
|
|
{
|
2015-03-19 11:23:18 +00:00
|
|
|
struct dispmanx_video *_dispvars = data;
|
2015-06-16 19:30:16 +00:00
|
|
|
int i, dst_width, dst_height, dst_xpos, dst_ypos;
|
2015-11-28 16:00:44 +00:00
|
|
|
struct dispmanx_surface *surface = NULL;
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
*sp = calloc (1, sizeof(struct dispmanx_surface));
|
2015-11-28 16:00:44 +00:00
|
|
|
|
|
|
|
surface = *sp;
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
/* Setup surface parameters */
|
|
|
|
surface->numpages = numpages;
|
|
|
|
/* We receive the pitch for what we consider "useful info",
|
|
|
|
* excluding things that are between scanlines. */
|
|
|
|
surface->pitch = visible_pitch;
|
|
|
|
|
|
|
|
/* Transparency disabled */
|
|
|
|
surface->alpha.flags = DISPMANX_FLAGS_ALPHA_FIXED_ALL_PIXELS;
|
|
|
|
surface->alpha.opacity = alpha;
|
|
|
|
surface->alpha.mask = 0;
|
2015-06-16 19:30:16 +00:00
|
|
|
|
2015-06-18 12:31:42 +00:00
|
|
|
/* Allocate memory for all the pages in each surface
|
|
|
|
* and initialize variables inside each page's struct. */
|
|
|
|
surface->pages = calloc(surface->numpages, sizeof(struct dispmanx_page));
|
2015-11-28 16:07:52 +00:00
|
|
|
|
|
|
|
for (i = 0; i < surface->numpages; i++)
|
|
|
|
{
|
2015-06-18 12:31:42 +00:00
|
|
|
surface->pages[i].used = false;
|
2015-06-20 14:56:25 +00:00
|
|
|
surface->pages[i].surface = surface;
|
2015-06-27 19:14:30 +00:00
|
|
|
surface->pages[i].dispvars = _dispvars;
|
2015-06-18 12:31:42 +00:00
|
|
|
surface->pages[i].page_used_mutex = slock_new();
|
|
|
|
}
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* The "visible" width obtained from the core pitch. We blit based on
|
|
|
|
* the "visible" width, for cores with things between scanlines. */
|
2015-06-27 19:14:30 +00:00
|
|
|
int visible_width = visible_pitch / (bpp / 8);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
dst_width = _dispvars->dispmanx_height * aspect;
|
|
|
|
dst_height = _dispvars->dispmanx_height;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
/* If we obtain a scaled image width that is bigger than the physical screen width,
|
2015-11-28 16:00:44 +00:00
|
|
|
* then we keep the physical screen width as our maximun width. */
|
2015-06-16 19:30:16 +00:00
|
|
|
if (dst_width > _dispvars->dispmanx_width)
|
|
|
|
dst_width = _dispvars->dispmanx_width;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
dst_xpos = (_dispvars->dispmanx_width - dst_width) / 2;
|
|
|
|
dst_ypos = (_dispvars->dispmanx_height - dst_height) / 2;
|
2015-03-19 11:23:18 +00:00
|
|
|
|
|
|
|
/* We configure the rects now. */
|
2015-06-16 19:30:16 +00:00
|
|
|
vc_dispmanx_rect_set(&surface->dst_rect, dst_xpos, dst_ypos, dst_width, dst_height);
|
2015-06-27 19:14:30 +00:00
|
|
|
vc_dispmanx_rect_set(&surface->bmp_rect, 0, 0, src_width, src_height);
|
|
|
|
vc_dispmanx_rect_set(&surface->src_rect, 0, 0, src_width << 16, src_height << 16);
|
2015-06-16 19:30:16 +00:00
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
for (i = 0; i < surface->numpages; i++)
|
|
|
|
{
|
2015-06-27 19:14:30 +00:00
|
|
|
surface->pages[i].resource = vc_dispmanx_resource_create(pixformat,
|
2015-11-28 16:00:44 +00:00
|
|
|
visible_width, src_height, &(_dispvars->vc_image_ptr));
|
2015-06-16 19:30:16 +00:00
|
|
|
}
|
2015-03-19 11:23:18 +00:00
|
|
|
/* Add element. */
|
|
|
|
_dispvars->update = vc_dispmanx_update_start(0);
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
surface->element = vc_dispmanx_element_add(
|
2015-11-28 16:00:44 +00:00
|
|
|
_dispvars->update,_dispvars->display, layer,
|
|
|
|
&surface->dst_rect, surface->pages[0].resource,
|
|
|
|
&surface->src_rect, DISPMANX_PROTECTION_NONE,
|
|
|
|
&surface->alpha, 0, (DISPMANX_TRANSFORM_T)0);
|
2015-03-19 11:23:18 +00:00
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
vc_dispmanx_update_submit_sync(_dispvars->update);
|
2015-06-16 19:30:16 +00:00
|
|
|
}
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
static void dispmanx_surface_update(void *data, const void *frame,
|
|
|
|
struct dispmanx_surface *surface)
|
2015-06-16 19:30:16 +00:00
|
|
|
{
|
|
|
|
struct dispmanx_video *_dispvars = data;
|
2015-11-28 16:00:44 +00:00
|
|
|
struct dispmanx_page *page = NULL;
|
|
|
|
|
2016-10-17 09:23:34 +00:00
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
|
|
|
|
if (settings->video.max_swapchain_images >= 3)
|
|
|
|
{
|
|
|
|
/* Wait until last issued flip completes to get a free page. Also,
|
|
|
|
* dispmanx doesn't support issuing more than one pageflip. */
|
|
|
|
slock_lock(_dispvars->pending_mutex);
|
|
|
|
if (_dispvars->pageflip_pending > 0)
|
|
|
|
scond_wait(_dispvars->vsync_condition, _dispvars->pending_mutex);
|
|
|
|
slock_unlock(_dispvars->pending_mutex);
|
|
|
|
}
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
page = dispmanx_get_free_page(_dispvars, surface);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
/* Frame blitting */
|
2015-06-16 19:30:16 +00:00
|
|
|
vc_dispmanx_resource_write_data(page->resource, surface->pixformat,
|
2015-11-28 16:00:44 +00:00
|
|
|
surface->pitch, (void*)frame, &(surface->bmp_rect));
|
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
/* Issue a page flip that will be done at the next vsync. */
|
|
|
|
_dispvars->update = vc_dispmanx_update_start(0);
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
vc_dispmanx_element_change_source(_dispvars->update, surface->element,
|
|
|
|
page->resource);
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
vc_dispmanx_update_submit(_dispvars->update, dispmanx_vsync_callback, (void*)page);
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
slock_lock(_dispvars->pending_mutex);
|
2016-10-17 09:23:34 +00:00
|
|
|
_dispvars->pageflip_pending++;
|
2015-03-19 11:23:18 +00:00
|
|
|
slock_unlock(_dispvars->pending_mutex);
|
2016-10-17 09:23:34 +00:00
|
|
|
|
|
|
|
if (settings->video.max_swapchain_images <= 2)
|
|
|
|
{
|
|
|
|
/* Wait for page flip before continuing, i.e. do not allow core to run
|
|
|
|
* ahead. This reduces input lag, but is less forgiving performance-
|
|
|
|
* wise. */
|
|
|
|
slock_lock(_dispvars->pending_mutex);
|
|
|
|
if (_dispvars->pageflip_pending > 0)
|
|
|
|
scond_wait(_dispvars->vsync_condition, _dispvars->pending_mutex);
|
|
|
|
slock_unlock(_dispvars->pending_mutex);
|
|
|
|
}
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
2016-06-16 09:37:02 +00:00
|
|
|
/* Enable/disable bilinear filtering. */
|
|
|
|
static void dispmanx_set_scaling (bool bilinear_filter)
|
|
|
|
{
|
|
|
|
if (bilinear_filter)
|
|
|
|
vc_gencmd_send( "%s", "scaling_kernel 0 -2 -6 -8 -10 -8 -3 2 18 50 82 119 155 187 213 227 227 213 187 155 119 82 50 18 2 -3 -8 -10 -8 -6 -2 0 0");
|
|
|
|
else
|
|
|
|
vc_gencmd_send( "%s", "scaling_kernel 0 0 0 0 0 0 0 0 1 1 1 1 255 255 255 255 255 255 255 255 1 1 1 1 0 0 0 0 0 0 0 0 1");
|
|
|
|
}
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
static void dispmanx_blank_console (void *data)
|
|
|
|
{
|
|
|
|
/* Note that a 2-pixels array is needed to accomplish console blanking because with 1-pixel
|
|
|
|
* only the write data function doesn't work well, so when we do the only resource
|
|
|
|
* change in the surface update function, we will be seeing a distorted console. */
|
|
|
|
struct dispmanx_video *_dispvars = data;
|
|
|
|
uint16_t image[2] = {0x0000, 0x0000};
|
|
|
|
float aspect = (float)_dispvars->dispmanx_width / (float)_dispvars->dispmanx_height;
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_setup(_dispvars,
|
2015-11-28 16:00:44 +00:00
|
|
|
2,
|
|
|
|
2,
|
|
|
|
4,
|
|
|
|
16,
|
|
|
|
VC_IMAGE_RGB565,
|
|
|
|
255,
|
|
|
|
aspect,
|
|
|
|
1,
|
|
|
|
-1,
|
|
|
|
&_dispvars->back_surface);
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_update(_dispvars, image, _dispvars->back_surface);
|
2015-06-16 19:30:16 +00:00
|
|
|
}
|
|
|
|
|
2015-03-13 04:59:52 +00:00
|
|
|
static void *dispmanx_gfx_init(const video_info_t *video,
|
|
|
|
const input_driver_t **input, void **input_data)
|
|
|
|
{
|
2015-03-19 11:23:18 +00:00
|
|
|
struct dispmanx_video *_dispvars = calloc(1, sizeof(struct dispmanx_video));
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
if (!_dispvars)
|
|
|
|
return NULL;
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
bcm_host_init();
|
|
|
|
_dispvars->display = vc_dispmanx_display_open(0 /* LCD */);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* If the console framebuffer has active overscan settings,
|
|
|
|
* the user must have overscan_scale=1 in config.txt to have
|
|
|
|
* the same size for both fb console and dispmanx. */
|
2015-11-28 16:00:44 +00:00
|
|
|
graphics_get_display_size(_dispvars->display,
|
|
|
|
&_dispvars->dispmanx_width, &_dispvars->dispmanx_height);
|
2015-06-16 19:30:16 +00:00
|
|
|
|
|
|
|
/* Setup surface parameters */
|
|
|
|
_dispvars->vc_image_ptr = 0;
|
|
|
|
_dispvars->pageflip_pending = 0;
|
2015-03-19 11:23:18 +00:00
|
|
|
_dispvars->menu_active = false;
|
2015-11-28 16:00:44 +00:00
|
|
|
_dispvars->rgb32 = video->rgb32;
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
/* It's very important that we set aspect here because the
|
|
|
|
* call seq when a core is loaded is gfx_init()->set_aspect()->gfx_frame()
|
|
|
|
* and we don't want the main surface to be setup in set_aspect()
|
|
|
|
* before we get to gfx_frame(). */
|
2016-06-15 12:28:13 +00:00
|
|
|
_dispvars->aspect_ratio = video_driver_get_aspect_ratio();
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
/* Initialize the rest of the mutexes and conditions. */
|
|
|
|
_dispvars->vsync_condition = scond_new();
|
2015-06-16 19:30:16 +00:00
|
|
|
_dispvars->vsync_cond_mutex = slock_new();
|
2015-03-19 11:23:18 +00:00
|
|
|
_dispvars->pending_mutex = slock_new();
|
2015-11-28 16:00:44 +00:00
|
|
|
_dispvars->core_width = 0;
|
|
|
|
_dispvars->core_height = 0;
|
|
|
|
_dispvars->menu_width = 0;
|
|
|
|
_dispvars->menu_height = 0;
|
|
|
|
|
|
|
|
_dispvars->main_surface = NULL;
|
|
|
|
_dispvars->menu_surface = NULL;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
if (input && input_data)
|
|
|
|
*input = NULL;
|
2016-06-16 09:37:02 +00:00
|
|
|
|
|
|
|
/* Enable/disable dispmanx bilinear filtering. */
|
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
dispmanx_set_scaling(settings->video.smooth);
|
2015-06-27 19:14:30 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
dispmanx_blank_console(_dispvars);
|
2015-03-19 11:23:18 +00:00
|
|
|
return _dispvars;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool dispmanx_gfx_frame(void *data, const void *frame, unsigned width,
|
2015-08-03 21:01:07 +00:00
|
|
|
unsigned height, uint64_t frame_count, unsigned pitch, const char *msg)
|
2015-03-13 04:59:52 +00:00
|
|
|
{
|
2015-03-15 04:41:29 +00:00
|
|
|
struct dispmanx_video *_dispvars = data;
|
2016-06-15 12:28:13 +00:00
|
|
|
float aspect = video_driver_get_aspect_ratio();
|
2015-03-15 04:41:29 +00:00
|
|
|
|
2016-06-15 12:28:13 +00:00
|
|
|
if (width != _dispvars->core_width || height != _dispvars->core_height || _dispvars->aspect_ratio != aspect)
|
2015-03-13 21:39:18 +00:00
|
|
|
{
|
2015-05-09 13:41:10 +00:00
|
|
|
/* Sanity check. */
|
|
|
|
if (width == 0 || height == 0)
|
|
|
|
return true;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2016-06-15 12:28:13 +00:00
|
|
|
_dispvars->core_width = width;
|
|
|
|
_dispvars->core_height = height;
|
|
|
|
_dispvars->core_pitch = pitch;
|
|
|
|
_dispvars->aspect_ratio = aspect;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
|
|
|
if (_dispvars->main_surface != NULL)
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_free(_dispvars, &_dispvars->main_surface);
|
|
|
|
|
2016-06-15 12:28:13 +00:00
|
|
|
/* Internal resolution or ratio has changed, so we need
|
|
|
|
* to recreate the main surface. */
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_setup(_dispvars,
|
2015-11-28 16:00:44 +00:00
|
|
|
width,
|
|
|
|
height,
|
|
|
|
pitch,
|
|
|
|
_dispvars->rgb32 ? 32 : 16,
|
|
|
|
_dispvars->rgb32 ? VC_IMAGE_XRGB8888 : VC_IMAGE_RGB565,
|
|
|
|
255,
|
2016-06-15 12:28:13 +00:00
|
|
|
_dispvars->aspect_ratio,
|
2015-11-28 16:00:44 +00:00
|
|
|
3,
|
|
|
|
0,
|
|
|
|
&_dispvars->main_surface);
|
2016-06-15 12:28:13 +00:00
|
|
|
|
|
|
|
/* We need to recreate the menu surface too, if it exists already, so we
|
|
|
|
* free it and let dispmanx_set_texture_frame() recreate it as it detects it's NULL.*/
|
|
|
|
if (_dispvars->menu_active && _dispvars->menu_surface) {
|
|
|
|
dispmanx_surface_free(_dispvars, &_dispvars->menu_surface);
|
|
|
|
}
|
2015-03-15 04:41:29 +00:00
|
|
|
}
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-03-15 04:41:29 +00:00
|
|
|
if (_dispvars->menu_active)
|
|
|
|
{
|
2015-06-16 19:30:16 +00:00
|
|
|
char buf[128];
|
2015-03-15 04:41:29 +00:00
|
|
|
video_monitor_get_fps(buf, sizeof(buf), NULL, 0);
|
|
|
|
}
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* Update main surface: locate free page, blit and flip. */
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_update(_dispvars, frame, _dispvars->main_surface);
|
2015-03-15 04:41:29 +00:00
|
|
|
return true;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dispmanx_set_texture_enable(void *data, bool state, bool full_screen)
|
|
|
|
{
|
2015-03-13 21:39:18 +00:00
|
|
|
struct dispmanx_video *_dispvars = data;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
/* If it was active but it's not anymore... */
|
2015-11-28 16:00:44 +00:00
|
|
|
if (!state && _dispvars->menu_active)
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_free(_dispvars, &_dispvars->menu_surface);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
_dispvars->menu_active = state;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dispmanx_set_texture_frame(void *data, const void *frame, bool rgb32,
|
2015-11-28 16:00:44 +00:00
|
|
|
unsigned width, unsigned height, float alpha)
|
2015-03-13 04:59:52 +00:00
|
|
|
{
|
2015-03-13 21:39:18 +00:00
|
|
|
struct dispmanx_video *_dispvars = data;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
|
|
|
if (!_dispvars->menu_active)
|
|
|
|
return;
|
|
|
|
|
2016-06-15 12:28:13 +00:00
|
|
|
/* If menu is active in this frame but our menu surface is NULL, we allocate a new one.*/
|
2015-11-28 16:00:44 +00:00
|
|
|
if (_dispvars->menu_surface == NULL)
|
2015-03-13 21:39:18 +00:00
|
|
|
{
|
2015-11-28 16:00:44 +00:00
|
|
|
_dispvars->menu_width = width;
|
|
|
|
_dispvars->menu_height = height;
|
|
|
|
_dispvars->menu_pitch = width * (rgb32 ? 4 : 2);
|
|
|
|
|
|
|
|
dispmanx_surface_setup(_dispvars,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
_dispvars->menu_pitch,
|
2016-06-15 12:28:13 +00:00
|
|
|
16,
|
2015-11-28 16:00:44 +00:00
|
|
|
VC_IMAGE_RGBA16,
|
|
|
|
210,
|
2016-06-15 12:28:13 +00:00
|
|
|
_dispvars->aspect_ratio,
|
2015-11-28 16:00:44 +00:00
|
|
|
3,
|
|
|
|
0,
|
|
|
|
&_dispvars->menu_surface);
|
2015-03-13 21:39:18 +00:00
|
|
|
}
|
2015-11-28 16:00:44 +00:00
|
|
|
|
|
|
|
/* We update the menu surface if menu is active. */
|
|
|
|
dispmanx_surface_update(_dispvars, frame, _dispvars->menu_surface);
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dispmanx_gfx_set_nonblock_state(void *data, bool state)
|
|
|
|
{
|
2015-03-13 21:39:18 +00:00
|
|
|
struct dispmanx_video *vid = data;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-03-13 21:39:18 +00:00
|
|
|
(void)data;
|
|
|
|
(void)vid;
|
|
|
|
|
|
|
|
/* TODO */
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool dispmanx_gfx_alive(void *data)
|
|
|
|
{
|
2015-03-19 11:23:18 +00:00
|
|
|
(void)data;
|
|
|
|
return true; /* always alive */
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool dispmanx_gfx_focus(void *data)
|
|
|
|
{
|
2015-03-19 11:23:18 +00:00
|
|
|
(void)data;
|
|
|
|
return true; /* fb device always has focus */
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dispmanx_gfx_viewport_info(void *data, struct video_viewport *vp)
|
|
|
|
{
|
2015-11-28 16:00:44 +00:00
|
|
|
struct dispmanx_video *vid = data;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
if (!vid)
|
|
|
|
return;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
vp->x = vp->y = 0;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
vp->width = vp->full_width = vid->core_width;
|
|
|
|
vp->height = vp->full_height = vid->core_height;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool dispmanx_gfx_suppress_screensaver(void *data, bool enable)
|
|
|
|
{
|
2015-03-13 21:39:18 +00:00
|
|
|
(void)data;
|
|
|
|
(void)enable;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-03-13 21:39:18 +00:00
|
|
|
return false;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool dispmanx_gfx_has_windowed(void *data)
|
|
|
|
{
|
2015-03-13 21:39:18 +00:00
|
|
|
(void)data;
|
|
|
|
|
|
|
|
return false;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool dispmanx_gfx_set_shader(void *data,
|
2015-11-28 16:00:44 +00:00
|
|
|
enum rarch_shader_type type, const char *path)
|
2015-03-13 04:59:52 +00:00
|
|
|
{
|
2015-03-19 11:23:18 +00:00
|
|
|
(void)data;
|
|
|
|
(void)type;
|
|
|
|
(void)path;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
return false;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void dispmanx_gfx_set_rotation(void *data, unsigned rotation)
|
|
|
|
{
|
2015-03-19 11:23:18 +00:00
|
|
|
(void)data;
|
|
|
|
(void)rotation;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool dispmanx_gfx_read_viewport(void *data, uint8_t *buffer)
|
|
|
|
{
|
2015-03-19 11:23:18 +00:00
|
|
|
(void)data;
|
|
|
|
(void)buffer;
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-03-19 11:23:18 +00:00
|
|
|
return true;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
static void dispmanx_set_aspect_ratio (void *data, unsigned aspect_ratio_idx)
|
|
|
|
{
|
2016-06-15 12:28:13 +00:00
|
|
|
/* Due to RetroArch setting the data pointer to NULL internally
|
|
|
|
* on core change, data is going to be NULL here after we load
|
|
|
|
* a new core from the GUI, so we can't count on accessing it
|
|
|
|
* to store the aspect ratio we are going to use, so we tell RA
|
|
|
|
* to keep track of the new aspect ratio and we get it in gfx_frame()
|
|
|
|
* with video_driver_get_aspect_ratio() to find out if it has changed. */
|
|
|
|
|
|
|
|
switch (aspect_ratio_idx)
|
|
|
|
{
|
|
|
|
case ASPECT_RATIO_SQUARE:
|
|
|
|
video_driver_set_viewport_square_pixel();
|
|
|
|
break;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2016-06-15 12:28:13 +00:00
|
|
|
case ASPECT_RATIO_CORE:
|
|
|
|
video_driver_set_viewport_core();
|
|
|
|
break;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
2016-06-15 12:28:13 +00:00
|
|
|
case ASPECT_RATIO_CONFIG:
|
|
|
|
video_driver_set_viewport_config();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
2015-06-27 19:14:30 +00:00
|
|
|
}
|
2016-06-15 12:28:13 +00:00
|
|
|
|
|
|
|
video_driver_set_aspect_ratio_value(aspectratio_lut[aspect_ratio_idx].value);
|
2015-06-16 19:30:16 +00:00
|
|
|
}
|
|
|
|
|
2015-03-13 04:59:52 +00:00
|
|
|
static const video_poke_interface_t dispmanx_poke_interface = {
|
2015-12-20 19:52:23 +00:00
|
|
|
NULL,
|
|
|
|
NULL,
|
2015-05-09 13:52:06 +00:00
|
|
|
NULL, /* set_video_mode */
|
|
|
|
NULL, /* set_filtering */
|
|
|
|
NULL, /* get_video_output_size */
|
|
|
|
NULL, /* get_video_output_prev */
|
|
|
|
NULL, /* get_video_output_next */
|
|
|
|
NULL, /* get_current_framebuffer */
|
|
|
|
NULL, /* get_proc_address */
|
2015-06-16 19:30:16 +00:00
|
|
|
dispmanx_set_aspect_ratio,
|
2015-05-09 13:52:06 +00:00
|
|
|
NULL, /* dispmanx_apply_state_changes */
|
2015-03-13 04:59:52 +00:00
|
|
|
#ifdef HAVE_MENU
|
2015-05-09 13:52:06 +00:00
|
|
|
dispmanx_set_texture_frame,
|
|
|
|
dispmanx_set_texture_enable,
|
2015-03-13 04:59:52 +00:00
|
|
|
#endif
|
2015-05-09 13:52:06 +00:00
|
|
|
NULL, /* dispmanx_set_osd_msg */
|
|
|
|
NULL /* dispmanx_show_mouse */
|
2015-03-13 04:59:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void dispmanx_gfx_get_poke_interface(void *data,
|
2015-11-28 16:00:44 +00:00
|
|
|
const video_poke_interface_t **iface)
|
2015-03-13 04:59:52 +00:00
|
|
|
{
|
2015-03-13 21:39:18 +00:00
|
|
|
(void)data;
|
|
|
|
*iface = &dispmanx_poke_interface;
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
static void dispmanx_gfx_free(void *data)
|
|
|
|
{
|
|
|
|
struct dispmanx_video *_dispvars = data;
|
2015-11-28 16:00:44 +00:00
|
|
|
|
|
|
|
if (!_dispvars)
|
|
|
|
return;
|
|
|
|
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_free(_dispvars, &_dispvars->main_surface);
|
|
|
|
dispmanx_surface_free(_dispvars, &_dispvars->back_surface);
|
2015-11-28 16:00:44 +00:00
|
|
|
|
|
|
|
if (_dispvars->menu_surface)
|
2015-06-27 19:14:30 +00:00
|
|
|
dispmanx_surface_free(_dispvars, &_dispvars->menu_surface);
|
2015-03-13 21:39:18 +00:00
|
|
|
|
|
|
|
/* Close display and deinitialize. */
|
|
|
|
vc_dispmanx_display_close(_dispvars->display);
|
|
|
|
bcm_host_deinit();
|
|
|
|
|
|
|
|
/* Destroy mutexes and conditions. */
|
2015-03-15 04:23:13 +00:00
|
|
|
slock_free(_dispvars->pending_mutex);
|
2015-06-16 19:30:16 +00:00
|
|
|
slock_free(_dispvars->vsync_cond_mutex);
|
2015-03-15 04:23:13 +00:00
|
|
|
scond_free(_dispvars->vsync_condition);
|
2015-03-13 21:39:18 +00:00
|
|
|
|
2015-06-16 19:30:16 +00:00
|
|
|
free(_dispvars);
|
2015-03-13 04:59:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
video_driver_t video_dispmanx = {
|
2015-06-16 19:30:16 +00:00
|
|
|
dispmanx_gfx_init,
|
|
|
|
dispmanx_gfx_frame,
|
|
|
|
dispmanx_gfx_set_nonblock_state,
|
|
|
|
dispmanx_gfx_alive,
|
|
|
|
dispmanx_gfx_focus,
|
|
|
|
dispmanx_gfx_suppress_screensaver,
|
|
|
|
dispmanx_gfx_has_windowed,
|
|
|
|
dispmanx_gfx_set_shader,
|
|
|
|
dispmanx_gfx_free,
|
|
|
|
"dispmanx",
|
2015-04-26 17:46:59 +00:00
|
|
|
NULL, /* set_viewport */
|
2015-06-16 19:30:16 +00:00
|
|
|
dispmanx_gfx_set_rotation,
|
|
|
|
dispmanx_gfx_viewport_info,
|
|
|
|
dispmanx_gfx_read_viewport,
|
2015-03-16 08:33:16 +00:00
|
|
|
NULL, /* read_frame_raw */
|
2015-03-13 04:59:52 +00:00
|
|
|
|
2015-11-28 16:00:44 +00:00
|
|
|
#ifdef HAVE_OVERLAY
|
2015-06-16 19:30:16 +00:00
|
|
|
NULL, /* overlay_interface */
|
2015-11-28 16:00:44 +00:00
|
|
|
#endif
|
2015-06-16 19:30:16 +00:00
|
|
|
dispmanx_gfx_get_poke_interface
|
2015-03-13 04:59:52 +00:00
|
|
|
};
|