RetroArch/gfx/drivers/ps2_gfx.c

388 lines
9.8 KiB
C
Raw Normal View History

2018-09-18 08:08:06 +02:00
/* RetroArch - A frontend for libretro.
* Copyright (C) 2018 - Francisco Javier Trujillo Mata
2018-09-18 08:08:06 +02:00
*
* 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 "../video_driver.h"
2018-09-18 08:08:06 +02:00
#include "../../driver.h"
#include "../../verbosity.h"
2018-09-18 08:08:06 +02:00
#include <loadcore.h>
#include <kernel.h>
#include <gsKit.h>
#include <gsInline.h>
2018-09-18 08:08:06 +02:00
2018-10-15 00:48:22 +02:00
#define GS_WHITE GS_SETREG_RGBAQ(0xFF,0xFF,0xFF,0x00,0x00) // turn white GS Screen
2018-09-18 08:08:06 +02:00
typedef struct ps2_video
2018-09-18 08:08:06 +02:00
{
GSGLOBAL *gsGlobal;
2018-10-31 19:45:43 +01:00
GSTEXTURE *menuTexture;
GSTEXTURE *coreTexture;
2018-10-31 19:45:43 +01:00
bool menuVisible;
bool full_screen;
bool rgb32;
int menu_filter;
int core_filter;
} ps2_video_t;
2018-09-18 08:08:06 +02:00
// PRIVATE METHODS
2018-11-01 13:56:35 +01:00
static GSGLOBAL *init_GSGlobal(void) {
GSGLOBAL *gsGlobal = gsKit_init_global()
2018-09-18 08:08:06 +02:00
2018-11-01 13:56:35 +01:00
gsGlobal->PSM = GS_PSM_CT16;
// gsGlobal->PSMZ = GS_PSMZ_16S;
gsGlobal->DoubleBuffering = GS_SETTING_OFF;
gsGlobal->ZBuffering = GS_SETTING_OFF;
gsGlobal->PrimAlphaEnable = GS_SETTING_ON; /* Enable alpha blending for primitives. */
2018-09-18 08:08:06 +02:00
2018-10-15 00:48:22 +02:00
dmaKit_init(D_CTRL_RELE_OFF,D_CTRL_MFD_OFF, D_CTRL_STS_UNSPEC,
D_CTRL_STD_OFF, D_CTRL_RCYC_8, 1 << DMA_CHANNEL_GIF);
2018-09-18 08:08:06 +02:00
2018-10-15 00:48:22 +02:00
// Initialize the DMAC
dmaKit_chan_init(DMA_CHANNEL_GIF);
2018-09-18 08:08:06 +02:00
2018-11-01 13:56:35 +01:00
gsKit_init_screen(gsGlobal);
gsKit_mode_switch(gsGlobal, GS_ONESHOT);
2018-09-18 08:08:06 +02:00
2018-11-01 13:56:35 +01:00
gsKit_clear(gsGlobal, GS_WHITE);
2018-10-15 00:48:22 +02:00
2018-11-01 13:56:35 +01:00
return gsGlobal;
}
2018-09-18 08:08:06 +02:00
2018-11-01 13:56:35 +01:00
static void init_ps2_video(ps2_video_t *ps2) {
ps2->gsGlobal = init_GSGlobal();
ps2->menuTexture = malloc(sizeof *ps2->menuTexture);
ps2->coreTexture = malloc(sizeof *ps2->coreTexture);
}
2018-09-18 08:08:06 +02:00
static void deinitTexture(GSTEXTURE *texture) {
2018-11-01 13:56:35 +01:00
free(texture->Mem);
free(texture->Clut);
texture->Mem = NULL;
texture->Clut = NULL;
}
2018-09-18 08:08:06 +02:00
2018-11-01 13:56:35 +01:00
static void color_correction32(uint32_t *buffer, uint32_t dimensions)
{
2018-11-01 13:56:35 +01:00
uint32_t i;
uint32_t x32;
for (i = 0; i < dimensions; i++) {
2018-11-01 13:56:35 +01:00
x32 = buffer[i];
buffer[i] = ((x32 >> 16) & 0xFF) | ((x32 << 16) & 0xFF0000) | (x32 & 0xFF00FF00);
}
}
2018-11-01 13:56:35 +01:00
static void color_correction16(uint16_t *buffer, uint32_t dimensions)
{
2018-11-01 13:56:35 +01:00
uint32_t i;
uint16_t x16;
for (i = 0; i < dimensions; i++) {
2018-11-01 13:56:35 +01:00
x16 = buffer[i];
buffer[i] = (x16 & 0x8000) | ((x16 << 10) & 0x7C00) | (x16 & 0x3E0) | ((x16 >> 10) & 0x1F);
}
}
static void transfer_texture(GSTEXTURE *texture, const void *frame,
int width, int height, bool rgb32, int filter, bool color_correction) {
int PSM = rgb32 ? GS_PSM_CT32 : GS_PSM_CT16;
size_t size = gsKit_texture_size_ee(width, height, PSM);
if ( !texture->Mem ||
texture->Width != width ||
texture->Height != height ||
texture->PSM != PSM ) {
texture->Width = width;
texture->Height = height;
texture->PSM = PSM;
texture->Filter = filter;
free(texture->Mem);
texture->Mem = memalign(128, size);
}
if (color_correction) {
int pixels = width * height;
if (rgb32) {
2018-11-01 13:56:35 +01:00
uint32_t *buffer = (uint32_t *)frame;
color_correction32(buffer, pixels);
} else {
2018-11-01 13:56:35 +01:00
uint16_t *buffer = (uint16_t *)frame;
color_correction16(buffer, pixels);
}
}
memcpy(texture->Mem, frame, size);
}
2018-11-01 13:56:35 +01:00
static void vram_alloc(GSGLOBAL *gsGlobal, GSTEXTURE *texture) {
uint32_t size = gsKit_texture_size(texture->Width, texture->Height, texture->PSM);
texture->Vram=gsKit_vram_alloc(gsGlobal, size, GSKIT_ALLOC_USERBUFFER);
if(texture->Vram == GSKIT_ALLOC_ERROR) {
printf("VRAM Allocation Failed. Will not upload texture.\n");
}
}
static void prim_texture(GSGLOBAL *gsGlobal, GSTEXTURE *texture, int zPosition, bool full_screen) {
gsKit_prim_sprite_texture( gsGlobal, texture,
0.0f,
0.0f, // Y1
0.0f, // U1
0.0f, // V1
gsGlobal->Width, // X2
gsGlobal->Height, // Y2
texture->Width, // U2
texture->Height, // V2
zPosition,
GS_WHITE);
}
static void *ps2_gfx_init(const video_info_t *video,
const input_driver_t **input, void **input_data)
{
2018-10-17 19:02:50 +02:00
void *ps2input = NULL;
*input_data = NULL;
(void)video;
2018-09-18 08:08:06 +02:00
ps2_video_t *ps2 = (ps2_video_t*)calloc(1, sizeof(ps2_video_t));
2018-09-18 08:08:06 +02:00
2018-10-31 19:45:43 +01:00
if (!ps2)
return NULL;
2018-11-01 13:56:35 +01:00
init_ps2_video(ps2);
ps2->rgb32 = video->rgb32;
ps2->core_filter = video->smooth ? GS_FILTER_LINEAR : GS_FILTER_NEAREST;
2018-10-17 19:02:50 +02:00
if (input && input_data)
{
settings_t *settings = config_get_ptr();
ps2input = input_ps2.init(settings->arrays.input_joypad_driver);
*input = ps2input ? &input_ps2 : NULL;
*input_data = ps2input;
}
2018-09-18 08:08:06 +02:00
return ps2;
}
static bool ps2_gfx_frame(void *data, const void *frame,
2018-09-18 08:08:06 +02:00
unsigned width, unsigned height, uint64_t frame_count,
unsigned pitch, const char *msg, video_frame_info_t *video_info)
{
2018-10-10 08:02:13 +02:00
#ifdef DISPLAY_FPS
uint32_t diff;
static uint64_t currentTick,lastTick;
static int frames;
static float fps = 0.0;
#endif
ps2_video_t *ps2 = (ps2_video_t*)data;
if (!width || !height)
return false;
2018-11-01 13:56:35 +01:00
if (frame_count%120==0) {
printf("ps2_gfx_frame %i\n", frame_count);
}
2018-10-31 19:45:43 +01:00
gsKit_vram_clear(ps2->gsGlobal);
if (frame) {
transfer_texture(ps2->coreTexture, frame, width, height, ps2->rgb32, ps2->core_filter, 1);
2018-11-01 13:56:35 +01:00
vram_alloc(ps2->gsGlobal, ps2->coreTexture);
2018-10-31 19:45:43 +01:00
gsKit_texture_upload(ps2->gsGlobal, ps2->coreTexture);
2018-11-01 13:56:35 +01:00
prim_texture(ps2->gsGlobal, ps2->coreTexture, 1, 1);
2018-10-31 19:45:43 +01:00
}
2018-11-01 13:56:35 +01:00
if (ps2->menuVisible) {
vram_alloc(ps2->gsGlobal, ps2->menuTexture);
2018-10-31 19:45:43 +01:00
gsKit_texture_upload(ps2->gsGlobal, ps2->menuTexture);
2018-11-01 13:56:35 +01:00
prim_texture(ps2->gsGlobal, ps2->menuTexture, 1, 1);
2018-10-31 19:45:43 +01:00
}
2018-10-15 00:48:22 +02:00
gsKit_sync_flip(ps2->gsGlobal);
gsKit_queue_exec(ps2->gsGlobal);
2018-10-10 08:02:13 +02:00
2018-09-18 08:08:06 +02:00
return true;
}
static void ps2_gfx_set_nonblock_state(void *data, bool toggle)
2018-09-18 08:08:06 +02:00
{
(void)data;
(void)toggle;
2018-09-18 08:08:06 +02:00
}
static bool ps2_gfx_alive(void *data)
2018-09-18 08:08:06 +02:00
{
(void)data;
return true;
}
static bool ps2_gfx_focus(void *data)
2018-09-18 08:08:06 +02:00
{
(void)data;
return true;
}
static bool ps2_gfx_suppress_screensaver(void *data, bool enable)
2018-09-18 08:08:06 +02:00
{
(void)data;
(void)enable;
return false;
}
static bool ps2_gfx_has_windowed(void *data)
2018-09-18 08:08:06 +02:00
{
(void)data;
return true;
2018-09-18 08:08:06 +02:00
}
static void ps2_gfx_free(void *data)
2018-09-18 08:08:06 +02:00
{
ps2_video_t *ps2 = (ps2_video_t*)data;
2018-10-31 19:45:43 +01:00
gsKit_clear(ps2->gsGlobal, GS_WHITE);
2018-11-01 13:56:35 +01:00
gsKit_vram_clear(ps2->gsGlobal);
2018-10-31 19:45:43 +01:00
deinitTexture(ps2->menuTexture);
deinitTexture(ps2->coreTexture);
2018-11-01 13:56:35 +01:00
gsKit_deinit_global(ps2->gsGlobal);
2018-09-18 08:08:06 +02:00
free(data);
2018-09-18 08:08:06 +02:00
}
static bool ps2_gfx_set_shader(void *data,
enum rarch_shader_type type, const char *path)
2018-09-18 08:08:06 +02:00
{
(void)data;
(void)type;
(void)path;
2018-09-18 08:08:06 +02:00
return false;
2018-09-18 08:08:06 +02:00
}
static void ps2_gfx_set_rotation(void *data,
unsigned rotation)
2018-09-18 08:08:06 +02:00
{
(void)data;
(void)rotation;
2018-09-18 08:08:06 +02:00
}
static void ps2_gfx_viewport_info(void *data,
struct video_viewport *vp)
2018-09-18 08:08:06 +02:00
{
(void)data;
(void)vp;
2018-09-18 08:08:06 +02:00
}
static bool ps2_gfx_read_viewport(void *data, uint8_t *buffer, bool is_idle)
2018-09-18 08:08:06 +02:00
{
(void)data;
(void)buffer;
return true;
2018-09-18 08:08:06 +02:00
}
2018-10-10 08:02:13 +02:00
static void ps2_set_filtering(void *data, unsigned index, bool smooth)
{
ps2_video_t *ps2 = (ps2_video_t*)data;
ps2->menu_filter = smooth ? GS_FILTER_LINEAR : GS_FILTER_NEAREST;
2018-10-10 08:02:13 +02:00
}
static void ps2_set_aspect_ratio(void *data, unsigned aspect_ratio_idx)
{
ps2_video_t *ps2 = (ps2_video_t*)data;
}
static void ps2_apply_state_changes(void *data)
{
ps2_video_t *ps2 = (ps2_video_t*)data;
}
static void ps2_set_texture_frame(void *data, const void *frame, bool rgb32,
unsigned width, unsigned height, float alpha)
{
ps2_video_t *ps2 = (ps2_video_t*)data;
transfer_texture(ps2->menuTexture, frame, width, height, rgb32, ps2->menu_filter, 0);
2018-10-10 08:02:13 +02:00
}
2018-10-31 19:45:43 +01:00
static void ps2_set_texture_enable(void *data, bool enable, bool full_screen)
2018-10-10 08:02:13 +02:00
{
(void) full_screen;
ps2_video_t *ps2 = (ps2_video_t*)data;
2018-10-31 19:45:43 +01:00
ps2->menuVisible = enable;
ps2->full_screen = full_screen;
2018-10-10 08:02:13 +02:00
}
static const video_poke_interface_t ps2_poke_interface = {
NULL, /* get_flags */
NULL, /* set_coords */
NULL, /* set_mvp */
NULL,
NULL,
NULL,
NULL, /* get_refresh_rate */
ps2_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 */
ps2_set_aspect_ratio,
ps2_apply_state_changes,
ps2_set_texture_frame,
ps2_set_texture_enable,
NULL, /* set_osd_msg */
NULL, /* show_mouse */
NULL, /* grab_mouse_toggle */
NULL, /* get_current_shader */
NULL, /* get_current_software_framebuffer */
NULL /* get_hw_render_interface */
};
static void ps2_gfx_get_poke_interface(void *data,
const video_poke_interface_t **iface)
2018-09-18 08:08:06 +02:00
{
(void)data;
2018-10-10 08:02:13 +02:00
*iface = &ps2_poke_interface;
2018-09-18 08:08:06 +02:00
}
video_driver_t video_ps2 = {
ps2_gfx_init,
ps2_gfx_frame,
ps2_gfx_set_nonblock_state,
ps2_gfx_alive,
ps2_gfx_focus,
ps2_gfx_suppress_screensaver,
ps2_gfx_has_windowed,
ps2_gfx_set_shader,
ps2_gfx_free,
2018-09-18 08:08:06 +02:00
"ps2",
NULL, /* set_viewport */
ps2_gfx_set_rotation,
ps2_gfx_viewport_info,
ps2_gfx_read_viewport,
2018-09-18 08:08:06 +02:00
NULL, /* read_frame_raw */
2018-09-18 08:08:06 +02:00
#ifdef HAVE_OVERLAY
NULL, /* overlay_interface */
2018-09-18 08:08:06 +02:00
#endif
ps2_gfx_get_poke_interface,
2018-09-18 08:08:06 +02:00
};