mirror of
https://github.com/CTCaer/RetroArch.git
synced 2025-01-18 00:32:46 +00:00
(gfx) Add OSMesa context
This commit is contained in:
parent
ab8471959e
commit
8752313961
@ -732,7 +732,10 @@ ifeq ($(HAVE_GL_CONTEXT), 1)
|
||||
LIBS += $(EGL_LIBS)
|
||||
endif
|
||||
endif
|
||||
|
||||
ifeq ($(HAVE_OSMESA), 1)
|
||||
OBJ += gfx/drivers_context/osmesa_ctx.o
|
||||
LIBS += -lOSMesa
|
||||
endif
|
||||
ifeq ($(HAVE_FFMPEG), 1)
|
||||
ifneq ($(C89_BUILD), 1)
|
||||
ifneq ($(HAVE_GLES), 1)
|
||||
|
409
gfx/drivers_context/osmesa_ctx.c
Normal file
409
gfx/drivers_context/osmesa_ctx.c
Normal file
@ -0,0 +1,409 @@
|
||||
/* RetroArch - A frontend for libretro.
|
||||
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
|
||||
* Copyright (C) 2011-2016 - Daniel De Matteis
|
||||
*
|
||||
* 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 "../../driver.h"
|
||||
#include "../../runloop.h"
|
||||
#include "../common/gl_common.h"
|
||||
|
||||
#include <GL/osmesa.h>
|
||||
#include <unistd.h>
|
||||
#include <errno.h>
|
||||
|
||||
#include <sys/stat.h>
|
||||
#include <fcntl.h>
|
||||
|
||||
#include <sys/socket.h>
|
||||
#include <sys/un.h>
|
||||
#include <poll.h>
|
||||
|
||||
#if (OSMESA_MAJOR_VERSION * 1000 + OSMESA_MINOR_VERSION) >= 11002
|
||||
#define HAVE_OSMESA_CREATE_CONTEXT_ATTRIBS 1
|
||||
#endif
|
||||
|
||||
#if (OSMESA_MAJOR_VERSION * 1000 + OSMESA_MINOR_VERSION) >= 3005
|
||||
#define HAVE_OSMESA_CREATE_CONTEXT_EXT 1
|
||||
#endif
|
||||
|
||||
static bool g_osmesa_profile = OSMESA_COMPAT_PROFILE;
|
||||
static int g_osmesa_major = 2;
|
||||
static int g_osmesa_minor = 1;
|
||||
static int g_osmesa_format = OSMESA_RGBA;
|
||||
static int g_osmesa_bpp = 4;
|
||||
static const char *g_osmesa_fifo = "/tmp/osmesa-retroarch.sock";
|
||||
|
||||
typedef struct gfx_osmesa_ctx_data
|
||||
{
|
||||
uint8_t *screen;
|
||||
int width;
|
||||
int height;
|
||||
int components;
|
||||
|
||||
int frame_count;
|
||||
OSMesaContext ctx;
|
||||
int socket;
|
||||
int client;
|
||||
} gfx_ctx_osmesa_data_t;
|
||||
|
||||
static void osmesa_fifo_open(gfx_ctx_osmesa_data_t *osmesa) {
|
||||
struct sockaddr_un saun, fsaun;
|
||||
|
||||
osmesa->socket = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||
|
||||
if (osmesa->socket < 0) {
|
||||
perror("[osmesa] socket()");
|
||||
return;
|
||||
}
|
||||
|
||||
saun.sun_family = AF_UNIX;
|
||||
|
||||
strcpy(saun.sun_path, g_osmesa_fifo);
|
||||
|
||||
unlink(g_osmesa_fifo);
|
||||
|
||||
if (bind(osmesa->socket, &saun, sizeof(saun.sun_family) + sizeof(saun.sun_path)) < 0) {
|
||||
perror("[osmesa] bind()");
|
||||
close(osmesa->socket);
|
||||
return;
|
||||
}
|
||||
|
||||
if (listen(osmesa->socket, 1) < 0) {
|
||||
perror("[osmesa] listen()");
|
||||
close(osmesa->socket);
|
||||
return;
|
||||
}
|
||||
|
||||
RARCH_WARN("[osmesa] Frame size is %ix%i\n", osmesa->width, osmesa->height);
|
||||
RARCH_WARN("[osmesa] Please connect to unix:%s\n", g_osmesa_fifo);
|
||||
}
|
||||
|
||||
static void osmesa_fifo_accept(gfx_ctx_osmesa_data_t *osmesa) {
|
||||
int res;
|
||||
struct pollfd fds;
|
||||
fds.fd = osmesa->socket;
|
||||
fds.events = POLLIN;
|
||||
|
||||
if (osmesa->client)
|
||||
return;
|
||||
|
||||
res = poll(&fds, 1, 0);
|
||||
|
||||
if (res < 0)
|
||||
perror("[osmesa] poll");
|
||||
else if (res > 0)
|
||||
osmesa->client = accept(osmesa->socket, NULL, NULL);
|
||||
}
|
||||
|
||||
static void osmesa_fifo_write(gfx_ctx_osmesa_data_t *osmesa) {
|
||||
if (osmesa->client >= 0) {
|
||||
size_t len = osmesa->width * 4;
|
||||
|
||||
for (int i = osmesa->height -1; i >= 0; --i) {
|
||||
int res = write(osmesa->client, osmesa->screen + i * len, len);
|
||||
|
||||
if (res < 0)
|
||||
osmesa->client = -1;
|
||||
}
|
||||
|
||||
// int res = write(osmesa->client, osmesa->screen, osmesa->width * osmesa->height * 4);
|
||||
|
||||
// if (res < 0)
|
||||
// osmesa->client = 0;
|
||||
}
|
||||
}
|
||||
|
||||
static void *osmesa_ctx_init(void *video_driver)
|
||||
{
|
||||
#ifdef HAVE_OSMESA_CREATE_CONTEXT_ATTRIBS
|
||||
const int attribs[] = {
|
||||
OSMESA_FORMAT, g_osmesa_format,
|
||||
OSMESA_DEPTH_BITS, 0,
|
||||
OSMESA_STENCIL_BITS, 0,
|
||||
OSMESA_ACCUM_BITS, 0,
|
||||
OSMESA_PROFILE, g_osmesa_profile,
|
||||
OSMESA_CONTEXT_MAJOR_VERSION, g_osmesa_major,
|
||||
OSMESA_CONTEXT_MINOR_VERSION, g_osmesa_minor,
|
||||
0, 0
|
||||
};
|
||||
#endif
|
||||
|
||||
gfx_ctx_osmesa_data_t *osmesa = (gfx_ctx_osmesa_data_t*)
|
||||
calloc(1, sizeof(gfx_ctx_osmesa_data_t));
|
||||
|
||||
if (!osmesa)
|
||||
return NULL;
|
||||
|
||||
#ifdef HAVE_OSMESA_CREATE_CONTEXT_ATTRIBS
|
||||
osmesa->ctx = OSMesaCreateContextAttribs(attribs, NULL);
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_OSMESA_CREATE_CONTEXT_EXT
|
||||
if (!osmesa->ctx) {
|
||||
osmesa->ctx = OSMesaCreateContextExt(g_osmesa_format, 0, 0, 0, NULL);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!osmesa->ctx) {
|
||||
#if defined(HAVE_OSMESA_CREATE_CONTEXT_ATTRIBS) || defined(HAVE_OSMESA_CREATE_CONTEXT_EXT)
|
||||
RARCH_WARN("[osmesa]: Falling back to standard context creation.\n");
|
||||
#endif
|
||||
osmesa->ctx = OSMesaCreateContext(g_osmesa_format, NULL);
|
||||
} else
|
||||
fprintf(stderr, "\n\n\n\n\n\ ext worked \n\n\nn\n");
|
||||
|
||||
if (!osmesa->ctx) {
|
||||
free(osmesa);
|
||||
RARCH_WARN("[omesa]: Failed to initialize the context driver.\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return osmesa;
|
||||
}
|
||||
|
||||
static void osmesa_ctx_destroy(void *data)
|
||||
{
|
||||
gfx_ctx_osmesa_data_t *osmesa = (gfx_ctx_osmesa_data_t*)data;
|
||||
|
||||
if (!osmesa)
|
||||
return;
|
||||
|
||||
if (osmesa->socket)
|
||||
close(osmesa->socket);
|
||||
|
||||
unlink("/tmp/retroarch-osmesa.fifo");
|
||||
|
||||
free(osmesa->screen);
|
||||
OSMesaDestroyContext(osmesa->ctx);
|
||||
|
||||
free(osmesa);
|
||||
}
|
||||
|
||||
static bool osmesa_ctx_bind_api(void *data, enum gfx_ctx_api api, unsigned major,
|
||||
unsigned minor)
|
||||
{
|
||||
if (api != GFX_CTX_OPENGL_API)
|
||||
return false;
|
||||
|
||||
g_osmesa_profile = OSMESA_COMPAT_PROFILE;
|
||||
|
||||
if (major) {
|
||||
g_osmesa_major = major;
|
||||
g_osmesa_minor = minor;
|
||||
} else {
|
||||
g_osmesa_major = 2;
|
||||
g_osmesa_minor = 1;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static void osmesa_ctx_swap_interval(void *data, unsigned interval)
|
||||
{
|
||||
(void)data;
|
||||
(void)interval;
|
||||
}
|
||||
|
||||
static bool osmesa_ctx_set_video_mode(void *data, unsigned width, unsigned height,
|
||||
bool fullscreen)
|
||||
{
|
||||
gfx_ctx_osmesa_data_t *osmesa = (gfx_ctx_osmesa_data_t*)data;
|
||||
uint8_t *screen = osmesa->screen;
|
||||
|
||||
(void)fullscreen;
|
||||
|
||||
bool size_changed = (width * height) != (osmesa->width * osmesa->height);
|
||||
|
||||
if (!osmesa->screen || size_changed)
|
||||
screen = (uint8_t*)calloc(1, (width * height) * 4);
|
||||
|
||||
if (!screen)
|
||||
return false;
|
||||
|
||||
if (!OSMesaMakeCurrent(osmesa->ctx, screen, GL_UNSIGNED_BYTE, width, height)) {
|
||||
if (screen != osmesa->screen)
|
||||
free(screen);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
osmesa->width = width;
|
||||
osmesa->height = height;
|
||||
|
||||
if (osmesa->screen && osmesa->screen != screen)
|
||||
free(osmesa->screen);
|
||||
|
||||
osmesa->screen = screen;
|
||||
|
||||
if (!osmesa->socket) {
|
||||
// unlink(g_osmesa_fifo);
|
||||
// if (mkfifo(g_osmesa_fifo, 0666) == 0) {
|
||||
// RARCH_WARN("[osmesa]: Please connect the sink to the fifo...\n");
|
||||
// RARCH_WARN("[osmesa]: Picture size is %ux%u\n", width, height);
|
||||
// osmesa->socket = open(g_osmesa_fifo, O_WRONLY);
|
||||
|
||||
// if (osmesa->socket)
|
||||
// RARCH_WARN("[osmesa]: Initialized fifo at %s\n", g_osmesa_fifo);
|
||||
// }
|
||||
|
||||
// if (!osmesa->socket || osmesa->socket < 0) {
|
||||
// unlink(g_osmesa_fifo);
|
||||
// RARCH_WARN("[osmesa]: Failed to initialize fifo: %s\n", strerror(errno));
|
||||
// }
|
||||
osmesa_fifo_open(osmesa);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static void osmesa_ctx_get_video_size(void *data,
|
||||
unsigned *width, unsigned *height)
|
||||
{
|
||||
gfx_ctx_osmesa_data_t *osmesa = (gfx_ctx_osmesa_data_t*)data;
|
||||
|
||||
if (!osmesa)
|
||||
return;
|
||||
|
||||
*width = osmesa->width;
|
||||
*height = osmesa->height;
|
||||
}
|
||||
|
||||
static void osmesa_ctx_update_window_title(void *data)
|
||||
{
|
||||
static char buf[128] = {0};
|
||||
static char buf_fps[128] = {0};
|
||||
settings_t *settings = config_get_ptr();
|
||||
gfx_ctx_osmesa_data_t *osmesa = (gfx_ctx_osmesa_data_t*)data;
|
||||
|
||||
if (!osmesa)
|
||||
return;
|
||||
|
||||
*buf = *buf_fps = '\0';
|
||||
|
||||
video_monitor_get_fps(buf, sizeof(buf), buf_fps, sizeof(buf_fps));
|
||||
|
||||
if (settings->fps_show)
|
||||
runloop_msg_queue_push(buf_fps, 1, 1, false);
|
||||
}
|
||||
|
||||
static void osmesa_ctx_check_window(void *data, bool *quit, bool *resize,unsigned *width,
|
||||
unsigned *height, unsigned frame_count)
|
||||
{
|
||||
gfx_ctx_osmesa_data_t *osmesa = (gfx_ctx_osmesa_data_t*)data;
|
||||
|
||||
*width = osmesa->width;
|
||||
*height = osmesa->height;
|
||||
*resize = false;
|
||||
*quit = false;
|
||||
|
||||
osmesa->frame_count = frame_count;
|
||||
}
|
||||
|
||||
static bool osmesa_ctx_set_resize(void *data, unsigned width, unsigned height)
|
||||
{
|
||||
(void)data;
|
||||
(void)width;
|
||||
(void)height;
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool osmesa_ctx_has_focus(void *data)
|
||||
{
|
||||
(void)data;
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool osmesa_ctx_suppress_screensaver(void *data, bool enable)
|
||||
{
|
||||
(void)data;
|
||||
(void)enable;
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool osmesa_ctx_has_windowed(void *data)
|
||||
{
|
||||
(void)data;
|
||||
return true;
|
||||
}
|
||||
|
||||
static void osmesa_ctx_swap_buffers(void *data)
|
||||
{
|
||||
gfx_ctx_osmesa_data_t *osmesa = (gfx_ctx_osmesa_data_t*)data;
|
||||
osmesa_fifo_accept(osmesa);
|
||||
osmesa_fifo_write(osmesa);
|
||||
|
||||
// write(osmesa->socket, osmesa->screen, osmesa->width * osmesa->height * 4);
|
||||
}
|
||||
|
||||
static void osmesa_ctx_input_driver(void *data, const input_driver_t **input, void **input_data)
|
||||
{
|
||||
(void)data;
|
||||
*input = NULL;
|
||||
*input_data = NULL;
|
||||
}
|
||||
|
||||
static gfx_ctx_proc_t osmesa_ctx_get_proc_address(const char *name)
|
||||
{
|
||||
return (gfx_ctx_proc_t)OSMesaGetProcAddress(name);
|
||||
}
|
||||
|
||||
static void osmesa_ctx_show_mouse(void *data, bool state)
|
||||
{
|
||||
(void)data;
|
||||
(void)state;
|
||||
}
|
||||
|
||||
static uint32_t osmesa_ctx_get_flags(void *data)
|
||||
{
|
||||
uint32_t flags = 0;
|
||||
BIT32_SET(flags, GFX_CTX_FLAGS_NONE);
|
||||
(void)data;
|
||||
return flags;
|
||||
}
|
||||
|
||||
static void osmesa_ctx_set_flags(void *data, uint32_t flags)
|
||||
{
|
||||
(void)data;
|
||||
}
|
||||
|
||||
const gfx_ctx_driver_t gfx_ctx_osmesa =
|
||||
{
|
||||
osmesa_ctx_init,
|
||||
osmesa_ctx_destroy,
|
||||
osmesa_ctx_bind_api,
|
||||
osmesa_ctx_swap_interval,
|
||||
osmesa_ctx_set_video_mode,
|
||||
osmesa_ctx_get_video_size,
|
||||
NULL, /* get_video_output_size */
|
||||
NULL, /* get_video_output_prev */
|
||||
NULL, /* get_video_output_next */
|
||||
NULL, /* get_metrics */
|
||||
NULL, /* translate_aspect */
|
||||
osmesa_ctx_update_window_title,
|
||||
osmesa_ctx_check_window,
|
||||
osmesa_ctx_set_resize,
|
||||
osmesa_ctx_has_focus,
|
||||
osmesa_ctx_suppress_screensaver,
|
||||
osmesa_ctx_has_windowed,
|
||||
osmesa_ctx_swap_buffers,
|
||||
osmesa_ctx_input_driver,
|
||||
osmesa_ctx_get_proc_address,
|
||||
NULL,
|
||||
NULL,
|
||||
osmesa_ctx_show_mouse,
|
||||
"osmesa",
|
||||
osmesa_ctx_get_flags,
|
||||
osmesa_ctx_set_flags,
|
||||
NULL /* bind_hw_render */
|
||||
};
|
@ -78,6 +78,9 @@ static const gfx_ctx_driver_t *gfx_ctx_drivers[] = {
|
||||
#if (defined(HAVE_SDL) || defined(HAVE_SDL2)) && defined(HAVE_OPENGL)
|
||||
&gfx_ctx_sdl_gl,
|
||||
#endif
|
||||
#ifdef HAVE_OSMESA
|
||||
&gfx_ctx_osmesa,
|
||||
#endif
|
||||
#ifdef EMSCRIPTEN
|
||||
&gfx_ctx_emscripten,
|
||||
#endif
|
||||
|
@ -226,6 +226,7 @@ typedef struct gfx_ctx_ident
|
||||
const char *ident;
|
||||
} gfx_ctx_ident_t;
|
||||
|
||||
extern const gfx_ctx_driver_t gfx_ctx_osmesa;
|
||||
extern const gfx_ctx_driver_t gfx_ctx_sdl_gl;
|
||||
extern const gfx_ctx_driver_t gfx_ctx_x_egl;
|
||||
extern const gfx_ctx_driver_t gfx_ctx_wayland;
|
||||
|
@ -270,6 +270,8 @@ if [ "$HAVE_OPENGL" != 'no' ] && [ "$HAVE_GLES" != 'yes' ]; then
|
||||
|
||||
# fix undefined variables
|
||||
PKG_CONF_USED="$PKG_CONF_USED CG"
|
||||
|
||||
check_pkgconf OSMESA osmesa
|
||||
else
|
||||
echo "Notice: Ignoring Cg. Desktop OpenGL is not enabled."
|
||||
HAVE_CG='no'
|
||||
|
@ -88,3 +88,4 @@ HAVE_RJPEG=yes # RJPEG support
|
||||
HAVE_RTGA=yes # RTGA support
|
||||
HAVE_HID=yes # Low-level HID (Human Interface Device) support
|
||||
HAVE_LANGEXTRA=yes # Multi-language support
|
||||
HAVE_OSMESA=no # Off-screen Mesa rendering
|
||||
|
Loading…
x
Reference in New Issue
Block a user