2015-04-01 22:14:13 +01:00
|
|
|
/* RetroArch - A frontend for libretro.
|
2015-04-02 01:43:28 +02:00
|
|
|
* Copyright (C) 2014-2015 - Ali Bouhlel
|
2015-04-01 22:14:13 +01: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 <3ds.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <malloc.h>
|
2015-04-05 20:40:35 +01:00
|
|
|
#include "ctr_gu.h"
|
2015-04-08 18:59:41 +01:00
|
|
|
#include "ctr_sprite_shader_shbin.h"
|
2015-04-01 22:14:13 +01:00
|
|
|
|
|
|
|
#include "../../general.h"
|
|
|
|
#include "../../driver.h"
|
|
|
|
#include "../video_viewport.h"
|
|
|
|
#include "../video_monitor.h"
|
|
|
|
|
|
|
|
#include "retroarch.h"
|
2015-04-08 18:59:41 +01:00
|
|
|
#include "performance.h"
|
|
|
|
#include "retro_inline.h"
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
#define CTR_TOP_FRAMEBUFFER_WIDTH 400
|
|
|
|
#define CTR_TOP_FRAMEBUFFER_HEIGHT 240
|
|
|
|
#define CTR_GPU_FRAMEBUFFER ((void*)0x1F119400)
|
|
|
|
#define CTR_GPU_DEPTHBUFFER ((void*)0x1F370800)
|
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
float v;
|
|
|
|
float u;
|
|
|
|
float y;
|
|
|
|
float x;
|
|
|
|
} ctr_scale_vector_t;
|
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
typedef struct
|
|
|
|
{
|
2015-04-08 18:59:41 +01:00
|
|
|
s16 x0, y0, x1, y1;
|
2015-04-05 20:40:35 +01:00
|
|
|
s16 u, v;
|
|
|
|
} ctr_vertex_t;
|
|
|
|
|
2015-04-01 22:14:13 +01:00
|
|
|
typedef struct ctr_video
|
|
|
|
{
|
2015-04-05 20:40:35 +01:00
|
|
|
struct
|
|
|
|
{
|
|
|
|
uint32_t* display_list;
|
|
|
|
int display_list_size;
|
|
|
|
void* texture_linear;
|
|
|
|
void* texture_swizzled;
|
|
|
|
int texture_width;
|
|
|
|
int texture_height;
|
2015-04-08 18:59:41 +01:00
|
|
|
ctr_scale_vector_t scale_vector;
|
2015-04-05 20:40:35 +01:00
|
|
|
ctr_vertex_t* frame_coords;
|
|
|
|
}menu;
|
|
|
|
|
|
|
|
uint32_t* display_list;
|
|
|
|
int display_list_size;
|
|
|
|
void* texture_linear;
|
|
|
|
void* texture_swizzled;
|
|
|
|
int texture_width;
|
|
|
|
int texture_height;
|
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
ctr_scale_vector_t scale_vector;
|
2015-04-05 20:40:35 +01:00
|
|
|
ctr_vertex_t* frame_coords;
|
|
|
|
|
|
|
|
DVLB_s* dvlb;
|
|
|
|
shaderProgram_s shader;
|
|
|
|
|
2015-09-14 00:19:38 +01:00
|
|
|
video_viewport_t vp;
|
|
|
|
|
2015-04-01 22:14:13 +01:00
|
|
|
bool rgb32;
|
|
|
|
bool vsync;
|
|
|
|
bool smooth;
|
|
|
|
bool menu_texture_enable;
|
|
|
|
unsigned rotation;
|
2015-09-14 00:19:38 +01:00
|
|
|
bool keep_aspect;
|
|
|
|
bool should_resize;
|
|
|
|
|
2015-09-28 14:54:48 +01:00
|
|
|
void* empty_framebuffer;
|
2015-04-01 22:14:13 +01:00
|
|
|
} ctr_video_t;
|
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
static INLINE void ctr_set_scale_vector(ctr_scale_vector_t* vec,
|
2015-04-11 08:39:37 +02:00
|
|
|
int viewport_width, int viewport_height,
|
|
|
|
int texture_width, int texture_height)
|
2015-04-01 22:14:13 +01:00
|
|
|
{
|
2015-04-08 18:59:41 +01:00
|
|
|
vec->x = -2.0 / viewport_width;
|
|
|
|
vec->y = -2.0 / viewport_height;
|
|
|
|
vec->u = 1.0 / texture_width;
|
|
|
|
vec->v = -1.0 / texture_height;
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
2015-09-14 00:19:38 +01:00
|
|
|
static INLINE void ctr_set_screen_coords(ctr_video_t * ctr)
|
|
|
|
{
|
|
|
|
if (ctr->rotation == 0)
|
|
|
|
{
|
|
|
|
ctr->frame_coords->x0 = ctr->vp.x;
|
|
|
|
ctr->frame_coords->y0 = ctr->vp.y;
|
|
|
|
ctr->frame_coords->x1 = ctr->vp.x + ctr->vp.width;
|
|
|
|
ctr->frame_coords->y1 = ctr->vp.y + ctr->vp.height;
|
|
|
|
}
|
|
|
|
else if (ctr->rotation == 1) /* 90° */
|
|
|
|
{
|
|
|
|
ctr->frame_coords->x1 = ctr->vp.x;
|
|
|
|
ctr->frame_coords->y0 = ctr->vp.y;
|
|
|
|
ctr->frame_coords->x0 = ctr->vp.x + ctr->vp.width;
|
|
|
|
ctr->frame_coords->y1 = ctr->vp.y + ctr->vp.height;
|
|
|
|
}
|
|
|
|
else if (ctr->rotation == 2) /* 180° */
|
|
|
|
{
|
|
|
|
ctr->frame_coords->x1 = ctr->vp.x;
|
|
|
|
ctr->frame_coords->y1 = ctr->vp.y;
|
|
|
|
ctr->frame_coords->x0 = ctr->vp.x + ctr->vp.width;
|
|
|
|
ctr->frame_coords->y0 = ctr->vp.y + ctr->vp.height;
|
|
|
|
}
|
|
|
|
else /* 270° */
|
|
|
|
{
|
|
|
|
ctr->frame_coords->x0 = ctr->vp.x;
|
|
|
|
ctr->frame_coords->y1 = ctr->vp.y;
|
|
|
|
ctr->frame_coords->x1 = ctr->vp.x + ctr->vp.width;
|
|
|
|
ctr->frame_coords->y0 = ctr->vp.y + ctr->vp.height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void ctr_update_viewport(ctr_video_t* ctr)
|
|
|
|
{
|
|
|
|
int x = 0;
|
|
|
|
int y = 0;
|
|
|
|
float device_aspect = ((float)ctr->vp.full_width) / ctr->vp.full_height;
|
|
|
|
float width = ctr->vp.full_width;
|
|
|
|
float height = ctr->vp.full_height;
|
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
|
|
|
|
if (settings->video.scale_integer)
|
|
|
|
{
|
|
|
|
video_viewport_get_scaled_integer(&ctr->vp, ctr->vp.full_width,
|
|
|
|
ctr->vp.full_height, video_driver_get_aspect_ratio(), ctr->keep_aspect);
|
|
|
|
width = ctr->vp.width;
|
|
|
|
height = ctr->vp.height;
|
|
|
|
}
|
|
|
|
else if (ctr->keep_aspect)
|
|
|
|
{
|
|
|
|
#if defined(HAVE_MENU)
|
|
|
|
if (settings->video.aspect_ratio_idx == ASPECT_RATIO_CUSTOM)
|
|
|
|
{
|
|
|
|
struct video_viewport *custom = video_viewport_get_custom();
|
|
|
|
|
|
|
|
if (custom)
|
|
|
|
{
|
|
|
|
x = custom->x;
|
|
|
|
y = custom->y;
|
|
|
|
width = custom->width;
|
|
|
|
height = custom->height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
2015-09-29 18:08:33 +02:00
|
|
|
float delta;
|
|
|
|
float desired_aspect = video_driver_get_aspect_ratio();
|
|
|
|
|
2015-09-14 00:19:38 +01:00
|
|
|
if (fabsf(device_aspect - desired_aspect) < 0.0001f)
|
|
|
|
{
|
|
|
|
/* If the aspect ratios of screen and desired aspect
|
|
|
|
* ratio are sufficiently equal (floating point stuff),
|
|
|
|
* assume they are actually equal.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
else if (device_aspect > desired_aspect)
|
|
|
|
{
|
|
|
|
delta = (desired_aspect / device_aspect - 1.0f)
|
|
|
|
/ 2.0f + 0.5f;
|
|
|
|
x = (int)roundf(width * (0.5f - delta));
|
|
|
|
width = (unsigned)roundf(2.0f * width * delta);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delta = (device_aspect / desired_aspect - 1.0f)
|
|
|
|
/ 2.0f + 0.5f;
|
|
|
|
y = (int)roundf(height * (0.5f - delta));
|
|
|
|
height = (unsigned)roundf(2.0f * height * delta);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ctr->vp.x = x;
|
|
|
|
ctr->vp.y = y;
|
|
|
|
ctr->vp.width = width;
|
|
|
|
ctr->vp.height = height;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ctr->vp.x = ctr->vp.y = 0;
|
|
|
|
ctr->vp.width = width;
|
|
|
|
ctr->vp.height = height;
|
|
|
|
}
|
|
|
|
|
|
|
|
ctr_set_screen_coords(ctr);
|
|
|
|
|
|
|
|
ctr->should_resize = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
static void* ctr_init(const video_info_t* video,
|
2015-04-11 08:39:37 +02:00
|
|
|
const input_driver_t** input, void** input_data)
|
2015-04-01 22:14:13 +01:00
|
|
|
{
|
2015-04-05 20:40:35 +01:00
|
|
|
void* ctrinput = NULL;
|
|
|
|
ctr_video_t* ctr = (ctr_video_t*)linearAlloc(sizeof(ctr_video_t));
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
if (!ctr)
|
|
|
|
return NULL;
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
// gfxInitDefault();
|
|
|
|
// gfxSet3D(false);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
memset(ctr, 0, sizeof(ctr_video_t));
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-09-14 00:19:38 +01:00
|
|
|
|
|
|
|
ctr->vp.x = 0;
|
|
|
|
ctr->vp.y = 0;
|
|
|
|
ctr->vp.width = CTR_TOP_FRAMEBUFFER_WIDTH;
|
|
|
|
ctr->vp.height = CTR_TOP_FRAMEBUFFER_HEIGHT;
|
|
|
|
ctr->vp.full_width = CTR_TOP_FRAMEBUFFER_WIDTH;
|
|
|
|
ctr->vp.full_height = CTR_TOP_FRAMEBUFFER_HEIGHT;
|
|
|
|
|
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
ctr->display_list_size = 0x40000;
|
|
|
|
ctr->display_list = linearAlloc(ctr->display_list_size * sizeof(uint32_t));
|
|
|
|
GPU_Reset(NULL, ctr->display_list, ctr->display_list_size);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
ctr->texture_width = 512;
|
|
|
|
ctr->texture_height = 512;
|
|
|
|
ctr->texture_linear =
|
|
|
|
linearMemAlign(ctr->texture_width * ctr->texture_height * sizeof(uint32_t), 128);
|
|
|
|
ctr->texture_swizzled =
|
|
|
|
linearMemAlign(ctr->texture_width * ctr->texture_height * sizeof(uint32_t), 128);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
ctr->frame_coords = linearAlloc(sizeof(ctr_vertex_t));
|
|
|
|
ctr->frame_coords->x0 = 0;
|
|
|
|
ctr->frame_coords->y0 = 0;
|
|
|
|
ctr->frame_coords->x1 = CTR_TOP_FRAMEBUFFER_WIDTH;
|
|
|
|
ctr->frame_coords->y1 = CTR_TOP_FRAMEBUFFER_HEIGHT;
|
|
|
|
ctr->frame_coords->u = CTR_TOP_FRAMEBUFFER_WIDTH;
|
|
|
|
ctr->frame_coords->v = CTR_TOP_FRAMEBUFFER_HEIGHT;
|
|
|
|
GSPGPU_FlushDataCache(NULL, (u8*)ctr->frame_coords, sizeof(ctr_vertex_t));
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
ctr->menu.texture_width = 512;
|
|
|
|
ctr->menu.texture_height = 512;
|
|
|
|
ctr->menu.texture_linear =
|
|
|
|
linearMemAlign(ctr->texture_width * ctr->texture_height * sizeof(uint16_t), 128);
|
|
|
|
ctr->menu.texture_swizzled =
|
|
|
|
linearMemAlign(ctr->texture_width * ctr->texture_height * sizeof(uint16_t), 128);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
ctr->menu.frame_coords = linearAlloc(sizeof(ctr_vertex_t));
|
|
|
|
|
|
|
|
ctr->menu.frame_coords->x0 = 40;
|
|
|
|
ctr->menu.frame_coords->y0 = 0;
|
|
|
|
ctr->menu.frame_coords->x1 = CTR_TOP_FRAMEBUFFER_WIDTH - 40;
|
|
|
|
ctr->menu.frame_coords->y1 = CTR_TOP_FRAMEBUFFER_HEIGHT;
|
|
|
|
ctr->menu.frame_coords->u = CTR_TOP_FRAMEBUFFER_WIDTH - 80;
|
|
|
|
ctr->menu.frame_coords->v = CTR_TOP_FRAMEBUFFER_HEIGHT;
|
|
|
|
GSPGPU_FlushDataCache(NULL, (u8*)ctr->menu.frame_coords, sizeof(ctr_vertex_t));
|
|
|
|
|
|
|
|
ctr_set_scale_vector(&ctr->scale_vector,
|
|
|
|
CTR_TOP_FRAMEBUFFER_WIDTH, CTR_TOP_FRAMEBUFFER_HEIGHT,
|
|
|
|
ctr->texture_width, ctr->texture_height);
|
|
|
|
ctr_set_scale_vector(&ctr->menu.scale_vector,
|
|
|
|
CTR_TOP_FRAMEBUFFER_WIDTH, CTR_TOP_FRAMEBUFFER_HEIGHT,
|
|
|
|
ctr->menu.texture_width, ctr->menu.texture_height);
|
|
|
|
|
|
|
|
ctr->dvlb = DVLB_ParseFile((u32*)ctr_sprite_shader_shbin, ctr_sprite_shader_shbin_size);
|
|
|
|
ctrGuSetVshGsh(&ctr->shader, ctr->dvlb, 2, 2);
|
|
|
|
shaderProgramUse(&ctr->shader);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPU_SetViewport(VIRT_TO_PHYS(CTR_GPU_DEPTHBUFFER),
|
|
|
|
VIRT_TO_PHYS(CTR_GPU_FRAMEBUFFER),
|
|
|
|
0, 0, CTR_TOP_FRAMEBUFFER_HEIGHT, CTR_TOP_FRAMEBUFFER_WIDTH);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
// GPU_SetViewport(NULL,
|
|
|
|
// VIRT_TO_PHYS(CTR_GPU_FRAMEBUFFER),
|
|
|
|
// 0, 0, CTR_TOP_FRAMEBUFFER_HEIGHT, CTR_TOP_FRAMEBUFFER_WIDTH);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPU_DepthMap(-1.0f, 0.0f);
|
|
|
|
GPU_SetFaceCulling(GPU_CULL_NONE);
|
|
|
|
GPU_SetStencilTest(false, GPU_ALWAYS, 0x00, 0xFF, 0x00);
|
2015-09-07 18:31:30 +01:00
|
|
|
GPU_SetStencilOp(GPU_STENCIL_KEEP, GPU_STENCIL_KEEP, GPU_STENCIL_KEEP);
|
2015-04-08 18:59:41 +01:00
|
|
|
GPU_SetBlendingColor(0, 0, 0, 0);
|
|
|
|
// GPU_SetDepthTestAndWriteMask(true, GPU_GREATER, GPU_WRITE_ALL);
|
|
|
|
GPU_SetDepthTestAndWriteMask(false, GPU_ALWAYS, GPU_WRITE_ALL);
|
|
|
|
// GPU_SetDepthTestAndWriteMask(true, GPU_ALWAYS, GPU_WRITE_ALL);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPUCMD_AddMaskedWrite(GPUREG_0062, 0x1, 0);
|
|
|
|
GPUCMD_AddWrite(GPUREG_0118, 0);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPU_SetAlphaBlending(GPU_BLEND_ADD, GPU_BLEND_ADD,
|
|
|
|
GPU_SRC_ALPHA, GPU_ONE_MINUS_SRC_ALPHA,
|
|
|
|
GPU_SRC_ALPHA, GPU_ONE_MINUS_SRC_ALPHA);
|
|
|
|
GPU_SetAlphaTest(false, GPU_ALWAYS, 0x00);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPU_SetTextureEnable(GPU_TEXUNIT0);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPU_SetTexEnv(0,
|
|
|
|
GPU_TEVSOURCES(GPU_TEXTURE0, GPU_PRIMARY_COLOR, 0),
|
|
|
|
GPU_TEVSOURCES(GPU_TEXTURE0, GPU_PRIMARY_COLOR, 0),
|
|
|
|
GPU_TEVOPERANDS(0, 0, 0),
|
|
|
|
GPU_TEVOPERANDS(0, 0, 0),
|
|
|
|
GPU_MODULATE, GPU_MODULATE,
|
|
|
|
0xFFFFFFFF);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPU_SetTexEnv(1, GPU_PREVIOUS,GPU_PREVIOUS, 0, 0, 0, 0, 0);
|
|
|
|
GPU_SetTexEnv(2, GPU_PREVIOUS,GPU_PREVIOUS, 0, 0, 0, 0, 0);
|
|
|
|
GPU_SetTexEnv(3, GPU_PREVIOUS,GPU_PREVIOUS, 0, 0, 0, 0, 0);
|
|
|
|
GPU_SetTexEnv(4, GPU_PREVIOUS,GPU_PREVIOUS, 0, 0, 0, 0, 0);
|
|
|
|
GPU_SetTexEnv(5, GPU_PREVIOUS,GPU_PREVIOUS, 0, 0, 0, 0, 0);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
ctrGuSetAttributeBuffers(2,
|
|
|
|
VIRT_TO_PHYS(ctr->menu.frame_coords),
|
|
|
|
CTRGU_ATTRIBFMT(GPU_SHORT, 4) << 0 |
|
|
|
|
CTRGU_ATTRIBFMT(GPU_SHORT, 2) << 4,
|
|
|
|
sizeof(ctr_vertex_t));
|
2015-04-05 20:40:35 +01:00
|
|
|
GPUCMD_Finalize();
|
2015-04-13 01:50:00 +01:00
|
|
|
ctrGuFlushAndRun(true);
|
2015-04-05 20:40:35 +01:00
|
|
|
gspWaitForEvent(GSPEVENT_P3D, false);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
|
|
|
if (input && input_data)
|
|
|
|
{
|
|
|
|
ctrinput = input_ctr.init();
|
|
|
|
*input = ctrinput ? &input_ctr : NULL;
|
|
|
|
*input_data = ctrinput;
|
|
|
|
}
|
|
|
|
|
2015-09-14 00:19:38 +01:00
|
|
|
ctr->keep_aspect = true;
|
|
|
|
ctr->should_resize = true;
|
2015-09-14 01:59:58 +01:00
|
|
|
ctr->smooth = true;
|
2015-09-14 00:19:38 +01:00
|
|
|
|
2015-09-28 14:54:48 +01:00
|
|
|
ctr->empty_framebuffer = linearAlloc(320 * 240 * 2);
|
|
|
|
memset(ctr->empty_framebuffer, 0, 320 * 240 * 2);
|
|
|
|
|
2015-09-28 20:09:07 +01:00
|
|
|
driver_set_refresh_rate((32730.0 * 8192.0) / 4481134.0);
|
|
|
|
|
2015-04-01 22:14:13 +01:00
|
|
|
return ctr;
|
|
|
|
}
|
2015-04-13 01:50:00 +01:00
|
|
|
|
2015-04-01 22:14:13 +01:00
|
|
|
static bool ctr_frame(void* data, const void* frame,
|
2015-08-03 23:01:07 +02:00
|
|
|
unsigned width, unsigned height,
|
|
|
|
uint64_t frame_count,
|
|
|
|
unsigned pitch, const char* msg)
|
2015-04-01 22:14:13 +01:00
|
|
|
{
|
2015-09-20 10:02:47 +02:00
|
|
|
uint32_t diff;
|
2015-04-02 04:00:26 +01:00
|
|
|
static uint64_t currentTick,lastTick;
|
2015-09-20 10:02:47 +02:00
|
|
|
ctr_video_t *ctr = (ctr_video_t*)data;
|
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
static float fps = 0.0;
|
2015-04-02 04:00:26 +01:00
|
|
|
static int total_frames = 0;
|
2015-09-20 10:02:47 +02:00
|
|
|
static int frames = 0;
|
|
|
|
static struct retro_perf_counter ctrframe_f = {0};
|
2015-09-28 14:54:48 +01:00
|
|
|
uint32_t state_tmp;
|
|
|
|
touchPosition state_tmp_touch;
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-13 11:26:02 +02:00
|
|
|
extern bool select_pressed;
|
|
|
|
|
2015-04-01 22:14:13 +01:00
|
|
|
if (!width || !height)
|
|
|
|
{
|
|
|
|
gspWaitForEvent(GSPEVENT_VBlank0, true);
|
2015-04-17 19:45:07 +01:00
|
|
|
return true;
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if(!aptMainLoop())
|
|
|
|
{
|
2015-04-13 11:26:02 +02:00
|
|
|
event_command(EVENT_CMD_QUIT);
|
2015-04-17 19:45:07 +01:00
|
|
|
return true;
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (select_pressed)
|
|
|
|
{
|
2015-04-13 11:26:02 +02:00
|
|
|
event_command(EVENT_CMD_QUIT);
|
2015-04-17 19:45:07 +01:00
|
|
|
return true;
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
2015-09-28 14:54:48 +01:00
|
|
|
state_tmp = hidKeysDown();
|
|
|
|
hidTouchRead(&state_tmp_touch);
|
|
|
|
if((state_tmp & KEY_TOUCH) && (state_tmp_touch.py < 120))
|
|
|
|
{
|
|
|
|
extern PrintConsole* currentConsole;
|
|
|
|
if ((u8*)currentConsole->frameBuffer == gfxBottomFramebuffers[0])
|
|
|
|
gfxBottomFramebuffers[0] = (u8*)ctr->empty_framebuffer;
|
|
|
|
else
|
|
|
|
gfxBottomFramebuffers[0] = (u8*)currentConsole->frameBuffer;
|
|
|
|
}
|
|
|
|
|
2015-04-17 19:45:07 +01:00
|
|
|
svcWaitSynchronization(gspEvents[GSPEVENT_P3D], 20000000);
|
|
|
|
svcClearEvent(gspEvents[GSPEVENT_P3D]);
|
|
|
|
svcWaitSynchronization(gspEvents[GSPEVENT_PPF], 20000000);
|
|
|
|
svcClearEvent(gspEvents[GSPEVENT_PPF]);
|
|
|
|
|
2015-04-02 04:00:26 +01:00
|
|
|
frames++;
|
2015-04-17 19:45:07 +01:00
|
|
|
|
|
|
|
if (ctr->vsync)
|
|
|
|
svcWaitSynchronization(gspEvents[GSPEVENT_VBlank0], U64_MAX);
|
|
|
|
|
|
|
|
svcClearEvent(gspEvents[GSPEVENT_VBlank0]);
|
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
currentTick = svcGetSystemTick();
|
2015-09-20 10:02:47 +02:00
|
|
|
diff = currentTick - lastTick;
|
2015-04-13 01:50:00 +01:00
|
|
|
if(diff > CTR_CPU_TICKS_PER_SECOND)
|
2015-04-02 04:00:26 +01:00
|
|
|
{
|
2015-04-13 01:50:00 +01:00
|
|
|
fps = (float)frames * ((float) CTR_CPU_TICKS_PER_SECOND / (float) diff);
|
2015-04-02 04:00:26 +01:00
|
|
|
lastTick = currentTick;
|
|
|
|
frames = 0;
|
|
|
|
}
|
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
printf("fps: %8.4f frames: %i\r", fps, total_frames++);
|
2015-04-17 19:45:07 +01:00
|
|
|
fflush(stdout);
|
2015-04-02 04:00:26 +01:00
|
|
|
|
2015-09-20 10:02:47 +02:00
|
|
|
rarch_perf_init(&ctrframe_f, "ctrframe_f");
|
|
|
|
retro_perf_start(&ctrframe_f);
|
2015-04-13 01:50:00 +01:00
|
|
|
|
2015-09-14 00:19:38 +01:00
|
|
|
if (ctr->should_resize)
|
|
|
|
ctr_update_viewport(ctr);
|
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
ctrGuSetMemoryFill(true, (u32*)CTR_GPU_FRAMEBUFFER, 0x00000000,
|
|
|
|
(u32*)(CTR_GPU_FRAMEBUFFER + CTR_TOP_FRAMEBUFFER_WIDTH * CTR_TOP_FRAMEBUFFER_HEIGHT * sizeof(uint32_t)),
|
|
|
|
0x201, (u32*)CTR_GPU_DEPTHBUFFER, 0x00000000,
|
|
|
|
(u32*)(CTR_GPU_DEPTHBUFFER + CTR_TOP_FRAMEBUFFER_WIDTH * CTR_TOP_FRAMEBUFFER_HEIGHT * sizeof(uint32_t)),
|
|
|
|
0x201);
|
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
GPUCMD_SetBufferOffset(0);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
if (width > ctr->texture_width)
|
|
|
|
width = ctr->texture_width;
|
|
|
|
if (height > ctr->texture_height)
|
|
|
|
height = ctr->texture_height;
|
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
if(frame)
|
2015-04-02 04:00:26 +01:00
|
|
|
{
|
2015-04-13 01:50:00 +01:00
|
|
|
if(((((u32)(frame)) >= 0x14000000 && ((u32)(frame)) < 0x1c000000)) /* frame in linear memory */
|
|
|
|
&& !((u32)frame & 0x7F) /* 128-byte aligned */
|
|
|
|
&& !((pitch) & 0xF)) /* 16-byte aligned */
|
2015-04-02 04:00:26 +01:00
|
|
|
{
|
2015-04-13 01:50:00 +01:00
|
|
|
/* can copy the buffer directly with the GPU */
|
|
|
|
ctrGuCopyImage(false, frame, pitch / 2, height, CTRGU_RGB565, false,
|
|
|
|
ctr->texture_swizzled, ctr->texture_width, CTRGU_RGB565, true);
|
2015-04-02 04:00:26 +01:00
|
|
|
}
|
2015-04-13 01:50:00 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
int i;
|
2015-09-20 10:02:47 +02:00
|
|
|
uint16_t *dst = (uint16_t*)ctr->texture_linear;
|
|
|
|
const uint8_t *src = frame;
|
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
for (i = 0; i < height; i++)
|
|
|
|
{
|
|
|
|
memcpy(dst, src, width * sizeof(uint16_t));
|
|
|
|
dst += ctr->texture_width;
|
|
|
|
src += pitch;
|
|
|
|
}
|
|
|
|
GSPGPU_FlushDataCache(NULL, ctr->texture_linear,
|
|
|
|
ctr->texture_width * ctr->texture_height * sizeof(uint16_t));
|
|
|
|
|
|
|
|
ctrGuCopyImage(false, ctr->texture_linear, ctr->texture_width, ctr->menu.texture_height, CTRGU_RGB565, false,
|
|
|
|
ctr->texture_swizzled, ctr->texture_width, CTRGU_RGB565, true);
|
2015-04-05 20:40:35 +01:00
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
}
|
2015-04-05 20:40:35 +01:00
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
}
|
2015-04-05 20:40:35 +01:00
|
|
|
|
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
ctrGuSetTexture(GPU_TEXUNIT0, VIRT_TO_PHYS(ctr->texture_swizzled), ctr->texture_width, ctr->texture_height,
|
2015-09-14 00:19:38 +01:00
|
|
|
(ctr->smooth? GPU_TEXTURE_MAG_FILTER(GPU_LINEAR) | GPU_TEXTURE_MIN_FILTER(GPU_LINEAR)
|
|
|
|
: GPU_TEXTURE_MAG_FILTER(GPU_NEAREST) | GPU_TEXTURE_MIN_FILTER(GPU_NEAREST)) |
|
2015-04-13 01:50:00 +01:00
|
|
|
GPU_TEXTURE_WRAP_S(GPU_CLAMP_TO_EDGE) | GPU_TEXTURE_WRAP_T(GPU_CLAMP_TO_EDGE),
|
|
|
|
GPU_RGB565);
|
2015-04-05 20:40:35 +01:00
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
ctr->frame_coords->u = width;
|
|
|
|
ctr->frame_coords->v = height;
|
|
|
|
GSPGPU_FlushDataCache(NULL, (u8*)ctr->frame_coords, sizeof(ctr_vertex_t));
|
2015-04-02 04:00:26 +01:00
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
ctrGuSetAttributeBuffersAddress(VIRT_TO_PHYS(ctr->frame_coords));
|
|
|
|
ctrGuSetVertexShaderFloatUniform(0, (float*)&ctr->scale_vector, 1);
|
2015-09-07 18:31:30 +01:00
|
|
|
GPU_DrawArray(GPU_UNKPRIM, 0, 1);
|
2015-04-02 04:00:26 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
if (ctr->menu_texture_enable)
|
|
|
|
{
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
GSPGPU_FlushDataCache(NULL, ctr->menu.texture_linear,
|
|
|
|
ctr->menu.texture_width * ctr->menu.texture_height * sizeof(uint16_t));
|
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
ctrGuCopyImage(false, ctr->menu.texture_linear, ctr->menu.texture_width, ctr->menu.texture_height, CTRGU_RGBA4444,false,
|
2015-04-05 20:40:35 +01:00
|
|
|
ctr->menu.texture_swizzled, ctr->menu.texture_width, CTRGU_RGBA4444, true);
|
|
|
|
|
|
|
|
ctrGuSetTexture(GPU_TEXUNIT0, VIRT_TO_PHYS(ctr->menu.texture_swizzled), ctr->menu.texture_width, ctr->menu.texture_height,
|
|
|
|
GPU_TEXTURE_MAG_FILTER(GPU_LINEAR) | GPU_TEXTURE_MIN_FILTER(GPU_LINEAR) |
|
|
|
|
GPU_TEXTURE_WRAP_S(GPU_CLAMP_TO_EDGE) | GPU_TEXTURE_WRAP_T(GPU_CLAMP_TO_EDGE),
|
|
|
|
GPU_RGBA4);
|
|
|
|
|
|
|
|
|
2015-04-08 18:59:41 +01:00
|
|
|
ctrGuSetAttributeBuffersAddress(VIRT_TO_PHYS(ctr->menu.frame_coords));
|
|
|
|
ctrGuSetVertexShaderFloatUniform(1, (float*)&ctr->menu.scale_vector, 1);
|
2015-09-07 18:31:30 +01:00
|
|
|
GPU_DrawArray(GPU_UNKPRIM, 0, 1);
|
2015-04-05 20:40:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
GPU_FinishDrawing();
|
|
|
|
GPUCMD_Finalize();
|
2015-04-13 01:50:00 +01:00
|
|
|
ctrGuFlushAndRun(true);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-13 01:50:00 +01:00
|
|
|
ctrGuDisplayTransfer(true, CTR_GPU_FRAMEBUFFER, 240,400, CTRGU_RGBA8,
|
2015-04-05 20:40:35 +01:00
|
|
|
gfxGetFramebuffer(GFX_TOP, GFX_LEFT, NULL, NULL), 240,400,CTRGU_RGB8, CTRGU_MULTISAMPLE_NONE);
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-09-13 22:27:28 +01:00
|
|
|
gfxSwapBuffersGpu();
|
2015-09-20 10:02:47 +02:00
|
|
|
retro_perf_stop(&ctrframe_f);
|
2015-05-09 15:41:10 +02:00
|
|
|
|
2015-04-01 22:14:13 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ctr_set_nonblock_state(void* data, bool toggle)
|
|
|
|
{
|
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
|
|
|
|
|
|
|
if (ctr)
|
|
|
|
ctr->vsync = !toggle;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ctr_alive(void* data)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ctr_focus(void* data)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ctr_suppress_screensaver(void* data, bool enable)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)enable;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ctr_has_windowed(void* data)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ctr_free(void* data)
|
|
|
|
{
|
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
|
|
|
|
|
|
|
if (!ctr)
|
|
|
|
return;
|
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
shaderProgramFree(&ctr->shader);
|
|
|
|
DVLB_Free(ctr->dvlb);
|
|
|
|
linearFree(ctr->display_list);
|
|
|
|
linearFree(ctr->texture_linear);
|
|
|
|
linearFree(ctr->texture_swizzled);
|
|
|
|
linearFree(ctr->frame_coords);
|
|
|
|
linearFree(ctr->menu.texture_linear);
|
|
|
|
linearFree(ctr->menu.texture_swizzled);
|
|
|
|
linearFree(ctr->menu.frame_coords);
|
2015-09-28 14:54:48 +01:00
|
|
|
linearFree(ctr->empty_framebuffer);
|
2015-04-05 20:40:35 +01:00
|
|
|
linearFree(ctr);
|
|
|
|
// gfxExit();
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ctr_set_texture_frame(void* data, const void* frame, bool rgb32,
|
|
|
|
unsigned width, unsigned height, float alpha)
|
|
|
|
{
|
2015-04-05 20:40:35 +01:00
|
|
|
int i;
|
2015-04-01 22:14:13 +01:00
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
2015-04-05 20:40:35 +01:00
|
|
|
uint16_t* dst = (uint16_t*)ctr->menu.texture_linear;
|
|
|
|
const uint16_t* src = frame;
|
|
|
|
int line_width = width;
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
(void)rgb32;
|
|
|
|
(void)alpha;
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
if (line_width > ctr->menu.texture_width)
|
|
|
|
line_width = ctr->menu.texture_width;
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
if (height > (unsigned)ctr->menu.texture_height)
|
|
|
|
height = (unsigned)ctr->menu.texture_height;
|
2015-04-01 22:14:13 +01:00
|
|
|
|
2015-04-05 20:40:35 +01:00
|
|
|
for (i = 0; i < height; i++)
|
|
|
|
{
|
|
|
|
memcpy(dst, src, line_width * sizeof(uint16_t));
|
|
|
|
dst += ctr->menu.texture_width;
|
|
|
|
src += width;
|
|
|
|
}
|
2015-04-08 18:59:41 +01:00
|
|
|
|
|
|
|
ctr->menu.frame_coords->x0 = (CTR_TOP_FRAMEBUFFER_WIDTH - width) / 2;
|
|
|
|
ctr->menu.frame_coords->y0 = (CTR_TOP_FRAMEBUFFER_HEIGHT - height) / 2;
|
|
|
|
ctr->menu.frame_coords->x1 = ctr->menu.frame_coords->x0 + width;
|
|
|
|
ctr->menu.frame_coords->y1 = ctr->menu.frame_coords->y0 + height;
|
|
|
|
ctr->menu.frame_coords->u = width;
|
|
|
|
ctr->menu.frame_coords->v = height;
|
|
|
|
GSPGPU_FlushDataCache(NULL, (u8*)ctr->menu.frame_coords, sizeof(ctr_vertex_t));
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ctr_set_texture_enable(void* data, bool state, bool full_screen)
|
|
|
|
{
|
|
|
|
(void) full_screen;
|
|
|
|
|
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
|
|
|
|
|
|
|
if (ctr)
|
|
|
|
ctr->menu_texture_enable = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void ctr_set_rotation(void* data, unsigned rotation)
|
|
|
|
{
|
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
|
|
|
|
|
|
|
if (!ctr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ctr->rotation = rotation;
|
2015-09-14 00:19:38 +01:00
|
|
|
ctr->should_resize = true;
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
static void ctr_set_filtering(void* data, unsigned index, bool smooth)
|
|
|
|
{
|
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
|
|
|
|
|
|
|
if (ctr)
|
|
|
|
ctr->smooth = smooth;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ctr_set_aspect_ratio(void* data, unsigned aspectratio_index)
|
|
|
|
{
|
2015-09-14 00:19:38 +01:00
|
|
|
ctr_video_t *ctr = (ctr_video_t*)data;
|
|
|
|
struct retro_system_av_info *av_info = video_viewport_get_system_av_info();
|
|
|
|
|
|
|
|
switch (aspectratio_index)
|
|
|
|
{
|
|
|
|
case ASPECT_RATIO_SQUARE:
|
|
|
|
video_viewport_set_square_pixel(
|
|
|
|
av_info->geometry.base_width,
|
|
|
|
av_info->geometry.base_height);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ASPECT_RATIO_CORE:
|
|
|
|
video_viewport_set_core();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ASPECT_RATIO_CONFIG:
|
|
|
|
video_viewport_set_config();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
video_driver_set_aspect_ratio_value(aspectratio_lut[aspectratio_index].value);
|
|
|
|
|
|
|
|
ctr->keep_aspect = true;
|
|
|
|
ctr->should_resize = true;
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ctr_apply_state_changes(void* data)
|
|
|
|
{
|
2015-09-14 00:19:38 +01:00
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
|
|
|
|
|
|
|
if (ctr)
|
|
|
|
ctr->should_resize = true;
|
|
|
|
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ctr_viewport_info(void* data, struct video_viewport* vp)
|
|
|
|
{
|
2015-09-14 00:19:38 +01:00
|
|
|
ctr_video_t* ctr = (ctr_video_t*)data;
|
|
|
|
|
|
|
|
if (ctr)
|
|
|
|
*vp = ctr->vp;
|
2015-04-01 22:14:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static const video_poke_interface_t ctr_poke_interface =
|
|
|
|
{
|
|
|
|
NULL,
|
|
|
|
ctr_set_filtering,
|
|
|
|
NULL, /* get_video_output_size */
|
|
|
|
NULL, /* get_video_output_prev */
|
|
|
|
NULL, /* get_video_output_next */
|
2015-04-25 02:35:06 +02:00
|
|
|
NULL, /* get_current_framebuffer */
|
2015-04-01 22:14:13 +01:00
|
|
|
NULL,
|
|
|
|
ctr_set_aspect_ratio,
|
|
|
|
ctr_apply_state_changes,
|
|
|
|
#ifdef HAVE_MENU
|
|
|
|
ctr_set_texture_frame,
|
|
|
|
ctr_set_texture_enable,
|
|
|
|
#endif
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static void ctr_get_poke_interface(void* data,
|
|
|
|
const video_poke_interface_t** iface)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
*iface = &ctr_poke_interface;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ctr_read_viewport(void* data, uint8_t* buffer)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)buffer;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ctr_set_shader(void* data,
|
|
|
|
enum rarch_shader_type type, const char* path)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)type;
|
|
|
|
(void)path;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
video_driver_t video_ctr =
|
|
|
|
{
|
|
|
|
ctr_init,
|
|
|
|
ctr_frame,
|
|
|
|
ctr_set_nonblock_state,
|
|
|
|
ctr_alive,
|
|
|
|
ctr_focus,
|
|
|
|
ctr_suppress_screensaver,
|
|
|
|
ctr_has_windowed,
|
|
|
|
ctr_set_shader,
|
|
|
|
ctr_free,
|
|
|
|
"ctr",
|
2015-04-26 19:46:59 +02:00
|
|
|
NULL, /* set_viewport */
|
2015-04-01 22:14:13 +01:00
|
|
|
ctr_set_rotation,
|
|
|
|
ctr_viewport_info,
|
|
|
|
ctr_read_viewport,
|
|
|
|
NULL, /* read_frame_raw */
|
|
|
|
#ifdef HAVE_OVERLAY
|
|
|
|
NULL,
|
|
|
|
#endif
|
|
|
|
ctr_get_poke_interface
|
|
|
|
};
|