2008-08-06 01:09:59 +00:00
/****************************************************************************
2008-09-12 05:28:40 +00:00
* Snes9x 1.51 Nintendo Wii / Gamecube Port
2008-08-06 01:09:59 +00:00
*
* softdev July 2006
* crunchy2 May 2007
2008-10-22 18:52:51 +00:00
* Michniewski 2008
* Tantric October 2008
2008-09-11 04:41:58 +00:00
*
* video . cpp
*
* Video routines
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
# include <gccore.h>
# include <ogcsys.h>
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
2008-08-06 01:39:43 +00:00
# include <unistd.h>
# include <wiiuse/wpad.h>
2009-01-16 04:48:15 +00:00
# include <ogc/texconv.h>
2008-08-06 01:09:59 +00:00
# include "snes9x.h"
# include "memmap.h"
# include "aram.h"
2008-08-06 02:31:44 +00:00
# include "snes9xGX.h"
2008-08-06 01:09:59 +00:00
2008-09-09 03:02:11 +00:00
# include "gui.h"
2008-08-06 01:09:59 +00:00
/*** Snes9x GFX Buffer ***/
static unsigned char snes9xgfx [ 1024 * 512 * 2 ] ;
/*** 2D Video ***/
2008-11-04 20:43:15 +00:00
unsigned int * xfb [ 2 ] = { NULL , NULL } ; // Double buffered
int whichfb = 0 ; // Switch
GXRModeObj * vmode ; // Menu video mode
2008-08-06 01:39:43 +00:00
int screenheight ;
2008-08-06 01:45:56 +00:00
extern u32 * backdrop ;
2008-08-06 01:09:59 +00:00
/*** GX ***/
# define TEX_WIDTH 512
# define TEX_HEIGHT 512
# define DEFAULT_FIFO_SIZE 256 * 1024
unsigned int copynow = GX_FALSE ;
static unsigned char gp_fifo [ DEFAULT_FIFO_SIZE ] ATTRIBUTE_ALIGN ( 32 ) ;
static unsigned char texturemem [ TEX_WIDTH * ( TEX_HEIGHT + 8 ) ] ATTRIBUTE_ALIGN ( 32 ) ;
GXTexObj texobj ;
2008-09-09 03:02:11 +00:00
Mtx view ;
2008-12-30 00:08:17 +00:00
static int vwidth , vheight , oldvwidth , oldvheight ;
2008-08-06 01:09:59 +00:00
u32 FrameTimer = 0 ;
2008-08-06 01:39:43 +00:00
u8 vmode_60hz = 0 ;
2008-09-11 02:23:48 +00:00
bool progressive = 0 ;
2008-08-06 01:39:43 +00:00
2008-08-23 23:15:58 +00:00
# define HASPECT 320
# define VASPECT 240
2008-08-06 01:09:59 +00:00
/* New texture based scaler */
typedef struct tagcamera
{
2008-11-04 20:43:15 +00:00
Vector pos ;
Vector up ;
Vector view ;
2008-08-06 01:09:59 +00:00
}
camera ;
/*** Square Matrix
This structure controls the size of the image on the screen .
Think of the output as a - 80 x 80 by - 60 x 60 graph .
* * */
s16 square [ ] ATTRIBUTE_ALIGN ( 32 ) =
{
/*
* X , Y , Z
* Values set are for roughly 4 : 3 aspect
*/
2008-11-04 20:43:15 +00:00
- HASPECT , VASPECT , 0 , // 0
HASPECT , VASPECT , 0 , // 1
HASPECT , - VASPECT , 0 , // 2
- HASPECT , - VASPECT , 0 // 3
2008-08-06 01:09:59 +00:00
} ;
2008-08-23 23:15:58 +00:00
2008-11-01 23:05:56 +00:00
static camera cam = {
{ 0.0F , 0.0F , 0.0F } ,
2008-11-04 20:43:15 +00:00
{ 0.0F , 0.5F , 0.0F } ,
{ 0.0F , 0.0F , - 0.5F }
2008-08-06 01:09:59 +00:00
} ;
2008-08-23 23:15:58 +00:00
2008-09-10 17:33:35 +00:00
/***
* * * Custom Video modes ( used to emulate original console video modes )
2008-08-23 23:15:58 +00:00
* * */
/** Original SNES PAL Resolutions: **/
/* 239 lines progressive (PAL 50Hz) */
2008-12-30 07:30:31 +00:00
static GXRModeObj TV_239p =
2008-08-23 23:15:58 +00:00
{
VI_TVMODE_PAL_DS , // viDisplayMode
2008-11-04 19:10:23 +00:00
512 , // fbWidth
2008-08-23 23:15:58 +00:00
239 , // efbHeight
239 , // xfbHeight
2008-09-10 21:09:53 +00:00
( VI_MAX_WIDTH_PAL - 640 ) / 2 , // viXOrigin
2008-11-04 19:10:23 +00:00
( VI_MAX_HEIGHT_PAL / 2 - 478 / 2 ) / 2 , // viYOrigin
2008-09-10 21:09:53 +00:00
640 , // viWidth
2008-09-09 16:24:42 +00:00
478 , // viHeight
2008-08-23 23:15:58 +00:00
VI_XFBMODE_SF , // xFBmode
GX_FALSE , // field_rendering
GX_FALSE , // aa
2008-11-04 20:43:15 +00:00
// sample points arranged in increasing Y order
{
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 0, 3 sample points, 1/12 units, 4 bits each
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 1
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 2
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } // pix 3
} ,
// vertical filter[7], 1/64 units, 6 bits each
{
0 , // line n-1
0 , // line n-1
21 , // line n
22 , // line n
21 , // line n
0 , // line n+1
0 // line n+1
}
2008-08-23 23:15:58 +00:00
} ;
/* 478 lines interlaced (PAL 50Hz, Deflicker) */
2008-12-30 07:30:31 +00:00
static GXRModeObj TV_478i =
2008-08-23 23:15:58 +00:00
{
2008-11-04 20:43:15 +00:00
VI_TVMODE_PAL_INT , // viDisplayMode
512 , // fbWidth
478 , // efbHeight
478 , // xfbHeight
( VI_MAX_WIDTH_PAL - 640 ) / 2 , // viXOrigin
( VI_MAX_HEIGHT_PAL - 478 ) / 2 , // viYOrigin
640 , // viWidth
478 , // viHeight
VI_XFBMODE_DF , // xFBmode
GX_FALSE , // field_rendering
GX_FALSE , // aa
// sample points arranged in increasing Y order
{
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 0, 3 sample points, 1/12 units, 4 bits each
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 1
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 2
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } // pix 3
} ,
// vertical filter[7], 1/64 units, 6 bits each
{
8 , // line n-1
8 , // line n-1
10 , // line n
12 , // line n
10 , // line n
8 , // line n+1
8 // line n+1
}
2008-08-23 23:15:58 +00:00
} ;
/** Original SNES NTSC Resolutions: **/
/* 224 lines progressive (NTSC or PAL 60Hz) */
2008-12-30 07:30:31 +00:00
static GXRModeObj TV_224p =
2008-08-23 23:15:58 +00:00
{
VI_TVMODE_EURGB60_DS , // viDisplayMode
2008-11-04 19:10:23 +00:00
512 , // fbWidth
2008-08-23 23:15:58 +00:00
224 , // efbHeight
224 , // xfbHeight
2008-09-10 21:09:53 +00:00
( VI_MAX_WIDTH_NTSC - 640 ) / 2 , // viXOrigin
2008-11-04 19:10:23 +00:00
( VI_MAX_HEIGHT_NTSC / 2 - 448 / 2 ) / 2 , // viYOrigin
2008-09-10 21:09:53 +00:00
640 , // viWidth
2008-09-09 16:24:42 +00:00
448 , // viHeight
2008-08-23 23:15:58 +00:00
VI_XFBMODE_SF , // xFBmode
GX_FALSE , // field_rendering
GX_FALSE , // aa
2008-11-04 20:43:15 +00:00
// sample points arranged in increasing Y order
{
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 0, 3 sample points, 1/12 units, 4 bits each
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 1
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 2
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } // pix 3
} ,
// vertical filter[7], 1/64 units, 6 bits each
{
0 , // line n-1
0 , // line n-1
21 , // line n
22 , // line n
21 , // line n
0 , // line n+1
0 // line n+1
}
2008-08-23 23:15:58 +00:00
} ;
/* 448 lines interlaced (NTSC or PAL 60Hz, Deflicker) */
2008-12-30 07:30:31 +00:00
static GXRModeObj TV_448i =
2008-08-23 23:15:58 +00:00
{
2008-11-04 20:43:15 +00:00
VI_TVMODE_EURGB60_INT , // viDisplayMode
512 , // fbWidth
448 , // efbHeight
448 , // xfbHeight
( VI_MAX_WIDTH_NTSC - 640 ) / 2 , // viXOrigin
( VI_MAX_HEIGHT_NTSC - 448 ) / 2 , // viYOrigin
640 , // viWidth
448 , // viHeight
VI_XFBMODE_DF , // xFBmode
GX_FALSE , // field_rendering
GX_FALSE , // aa
// sample points arranged in increasing Y order
{
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 0, 3 sample points, 1/12 units, 4 bits each
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 1
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } , // pix 2
{ 6 , 6 } , { 6 , 6 } , { 6 , 6 } // pix 3
} ,
// vertical filter[7], 1/64 units, 6 bits each
{
8 , // line n-1
8 , // line n-1
10 , // line n
12 , // line n
10 , // line n
8 , // line n+1
8 // line n+1
}
2008-08-23 23:15:58 +00:00
} ;
/* TV Modes table */
2008-12-30 07:30:31 +00:00
static GXRModeObj * tvmodes [ 4 ] = {
2008-08-23 23:15:58 +00:00
& TV_239p , & TV_478i , /* Snes PAL video modes */
& TV_224p , & TV_448i , /* Snes NTSC video modes */
} ;
2008-08-06 01:09:59 +00:00
/****************************************************************************
* VideoThreading
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
# define TSTACK 16384
2008-12-30 00:08:17 +00:00
static lwpq_t videoblankqueue ;
static lwp_t vbthread ;
2008-08-06 01:09:59 +00:00
static unsigned char vbstack [ TSTACK ] ;
/****************************************************************************
* vbgetback
*
* This callback enables the emulator to keep running while waiting for a
2008-08-10 03:14:39 +00:00
* vertical blank .
2008-08-06 01:09:59 +00:00
*
* Putting LWP to good use : )
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
static void *
vbgetback ( void * arg )
{
2008-08-23 23:15:58 +00:00
while ( 1 )
{
VIDEO_WaitVSync ( ) ; /**< Wait for video vertical blank */
LWP_SuspendThread ( vbthread ) ;
}
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
return NULL ;
2008-08-06 01:09:59 +00:00
}
/****************************************************************************
* InitVideoThread
*
* libOGC provides a nice wrapper for LWP access .
* This function sets up a new local queue and attaches the thread to it .
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
void
InitVideoThread ( )
{
/*** Initialise a new queue ***/
2008-08-23 23:15:58 +00:00
LWP_InitQueue ( & videoblankqueue ) ;
2008-08-06 01:09:59 +00:00
/*** Create the thread on this queue ***/
2008-12-19 23:07:57 +00:00
LWP_CreateThread ( & vbthread , vbgetback , NULL , vbstack , TSTACK , 150 ) ;
2008-08-06 01:09:59 +00:00
}
/****************************************************************************
* copy_to_xfb
*
* Stock code to copy the GX buffer to the current display mode .
* Also increments the frameticker , as it ' s called for each vb .
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
static void
copy_to_xfb ( u32 arg )
{
2008-08-23 23:15:58 +00:00
if ( copynow = = GX_TRUE )
{
GX_CopyDisp ( xfb [ whichfb ] , GX_TRUE ) ;
GX_Flush ( ) ;
copynow = GX_FALSE ;
}
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
FrameTimer + + ;
2008-08-06 01:09:59 +00:00
}
/****************************************************************************
2008-08-23 23:15:58 +00:00
* Scaler Support Functions
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
static void
draw_init ( )
{
2008-08-23 23:15:58 +00:00
GX_ClearVtxDesc ( ) ;
GX_SetVtxDesc ( GX_VA_POS , GX_INDEX8 ) ;
GX_SetVtxDesc ( GX_VA_CLR0 , GX_INDEX8 ) ;
GX_SetVtxDesc ( GX_VA_TEX0 , GX_DIRECT ) ;
GX_SetVtxAttrFmt ( GX_VTXFMT0 , GX_VA_POS , GX_POS_XYZ , GX_S16 , 0 ) ;
GX_SetVtxAttrFmt ( GX_VTXFMT0 , GX_VA_CLR0 , GX_CLR_RGBA , GX_RGBA8 , 0 ) ;
GX_SetVtxAttrFmt ( GX_VTXFMT0 , GX_VA_TEX0 , GX_TEX_ST , GX_F32 , 0 ) ;
GX_SetArray ( GX_VA_POS , square , 3 * sizeof ( s16 ) ) ;
GX_SetNumTexGens ( 1 ) ;
GX_SetNumChans ( 0 ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetTexCoordGen ( GX_TEXCOORD0 , GX_TG_MTX2x4 , GX_TG_TEX0 , GX_IDENTITY ) ;
2008-09-23 03:49:57 +00:00
2008-08-23 23:15:58 +00:00
GX_SetTevOp ( GX_TEVSTAGE0 , GX_REPLACE ) ;
GX_SetTevOrder ( GX_TEVSTAGE0 , GX_TEXCOORD0 , GX_TEXMAP0 , GX_COLORNULL ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
memset ( & view , 0 , sizeof ( Mtx ) ) ;
guLookAt ( view , & cam . pos , & cam . up , & cam . view ) ;
GX_LoadPosMtxImm ( view , GX_PNMTX0 ) ;
2008-11-04 19:10:23 +00:00
2008-11-04 20:43:15 +00:00
GX_InvVtxCache ( ) ; // update vertex cache
2008-08-06 01:09:59 +00:00
}
static void
draw_vert ( u8 pos , u8 c , f32 s , f32 t )
{
2008-08-23 23:15:58 +00:00
GX_Position1x8 ( pos ) ;
GX_Color1x8 ( c ) ;
GX_TexCoord2f32 ( s , t ) ;
2008-08-06 01:09:59 +00:00
}
static void
draw_square ( Mtx v )
{
2008-08-23 23:15:58 +00:00
Mtx m ; // model matrix.
Mtx mv ; // modelview matrix.
guMtxIdentity ( m ) ;
guMtxTransApply ( m , m , 0 , 0 , - 100 ) ;
guMtxConcat ( v , m , mv ) ;
GX_LoadPosMtxImm ( mv , GX_PNMTX0 ) ;
GX_Begin ( GX_QUADS , GX_VTXFMT0 , 4 ) ;
draw_vert ( 0 , 0 , 0.0 , 0.0 ) ;
draw_vert ( 1 , 0 , 1.0 , 0.0 ) ;
draw_vert ( 2 , 0 , 1.0 , 1.0 ) ;
draw_vert ( 3 , 0 , 0.0 , 1.0 ) ;
GX_End ( ) ;
2008-08-06 01:09:59 +00:00
}
/****************************************************************************
* StartGX
*
* This function initialises the GX .
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
static void
StartGX ( )
{
2008-12-30 07:30:31 +00:00
Mtx44 p ;
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
GXColor background = { 0 , 0 , 0 , 0xff } ;
2008-08-06 01:09:59 +00:00
/*** Clear out FIFO area ***/
2008-08-23 23:15:58 +00:00
memset ( & gp_fifo , 0 , DEFAULT_FIFO_SIZE ) ;
2008-08-06 01:09:59 +00:00
/*** Initialise GX ***/
2008-08-23 23:15:58 +00:00
GX_Init ( & gp_fifo , DEFAULT_FIFO_SIZE ) ;
GX_SetCopyClear ( background , 0x00ffffff ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetViewport ( 0 , 0 , vmode - > fbWidth , vmode - > efbHeight , 0 , 1 ) ;
GX_SetDispCopyYScale ( ( f32 ) vmode - > xfbHeight / ( f32 ) vmode - > efbHeight ) ;
GX_SetScissor ( 0 , 0 , vmode - > fbWidth , vmode - > efbHeight ) ;
GX_SetDispCopySrc ( 0 , 0 , vmode - > fbWidth , vmode - > efbHeight ) ;
GX_SetDispCopyDst ( vmode - > fbWidth , vmode - > xfbHeight ) ;
GX_SetCopyFilter ( vmode - > aa , vmode - > sample_pattern , GX_TRUE , vmode - > vfilter ) ;
GX_SetFieldMode ( vmode - > field_rendering , ( ( vmode - > viHeight = = 2 * vmode - > xfbHeight ) ? GX_ENABLE : GX_DISABLE ) ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetPixelFmt ( GX_PF_RGB8_Z24 , GX_ZC_LINEAR ) ;
GX_SetCullMode ( GX_CULL_NONE ) ;
GX_SetDispCopyGamma ( GX_GM_1_0 ) ;
2008-08-25 05:57:43 +00:00
GX_SetZMode ( GX_TRUE , GX_LEQUAL , GX_TRUE ) ;
2008-08-23 23:15:58 +00:00
GX_SetColorUpdate ( GX_TRUE ) ;
2008-09-10 17:33:35 +00:00
2008-09-09 03:02:11 +00:00
gui_alphasetup ( ) ;
2008-08-23 23:15:58 +00:00
guOrtho ( p , vmode - > efbHeight / 2 , - ( vmode - > efbHeight / 2 ) , - ( vmode - > fbWidth / 2 ) , vmode - > fbWidth / 2 , 10 , 1000 ) ; // matrix, t, b, l, r, n, f
GX_LoadProjectionMtx ( p , GX_ORTHOGRAPHIC ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_CopyDisp ( xfb [ whichfb ] , GX_TRUE ) ; // reset xfb
vwidth = 100 ;
vheight = 100 ;
2008-08-06 01:09:59 +00:00
}
2008-08-06 01:39:43 +00:00
/****************************************************************************
2008-08-23 23:15:58 +00:00
* UpdatePadsCB
2008-08-06 01:39:43 +00:00
*
* called by postRetraceCallback in InitGCVideo - scans gcpad and wpad
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-12-30 07:30:31 +00:00
static void
2008-08-23 23:15:58 +00:00
UpdatePadsCB ( )
2008-08-06 01:39:43 +00:00
{
2008-08-06 01:45:56 +00:00
# ifdef HW_RVL
2008-08-06 01:39:43 +00:00
WPAD_ScanPads ( ) ;
2008-08-06 01:45:56 +00:00
# endif
2008-08-06 01:39:43 +00:00
PAD_ScanPads ( ) ;
}
2008-08-10 03:14:39 +00:00
2008-08-06 01:09:59 +00:00
/****************************************************************************
* InitGCVideo
*
* This function MUST be called at startup .
2008-08-23 23:15:58 +00:00
* - also sets up menu video mode
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
void
InitGCVideo ( )
{
2008-08-23 23:15:58 +00:00
// init video
2008-08-06 01:09:59 +00:00
VIDEO_Init ( ) ;
2008-09-11 04:41:58 +00:00
2008-08-23 23:15:58 +00:00
// get default video mode
2008-08-06 01:39:43 +00:00
vmode = VIDEO_GetPreferredMode ( NULL ) ;
2008-08-23 23:15:58 +00:00
switch ( vmode - > viTVMode > > 2 )
2008-08-06 01:39:43 +00:00
{
2008-09-11 04:41:58 +00:00
case VI_PAL :
2008-09-11 02:23:48 +00:00
// 576 lines (PAL 50Hz)
// display should be centered vertically (borders)
2008-08-23 23:15:58 +00:00
vmode = & TVPal574IntDfScale ;
vmode - > xfbHeight = 480 ;
vmode - > viYOrigin = ( VI_MAX_HEIGHT_PAL - 480 ) / 2 ;
vmode - > viHeight = 480 ;
2008-09-11 02:23:48 +00:00
vmode_60hz = 0 ;
2008-11-04 19:10:23 +00:00
// Original Video modes (forced to PAL 50hz)
// set video signal mode
TV_224p . viTVMode = VI_TVMODE_PAL_DS ;
TV_448i . viTVMode = VI_TVMODE_PAL_INT ;
// set VI position
TV_224p . viYOrigin = ( VI_MAX_HEIGHT_PAL / 2 - 448 / 2 ) / 2 ;
TV_448i . viYOrigin = ( VI_MAX_HEIGHT_PAL - 448 ) / 2 ;
2008-08-06 01:39:43 +00:00
break ;
2008-09-11 04:41:58 +00:00
case VI_NTSC :
2008-09-11 02:23:48 +00:00
// 480 lines (NTSC 60hz)
2008-08-23 23:15:58 +00:00
vmode_60hz = 1 ;
2008-11-04 19:10:23 +00:00
// Original Video modes (forced to NTSC 60hz)
// set video signal mode
TV_239p . viTVMode = VI_TVMODE_NTSC_DS ;
TV_478i . viTVMode = VI_TVMODE_NTSC_INT ;
TV_224p . viTVMode = VI_TVMODE_NTSC_DS ;
TV_448i . viTVMode = VI_TVMODE_NTSC_INT ;
// set VI position
TV_239p . viYOrigin = ( VI_MAX_HEIGHT_NTSC / 2 - 478 / 2 ) / 2 ;
TV_478i . viYOrigin = ( VI_MAX_HEIGHT_NTSC - 478 ) / 2 ;
TV_224p . viYOrigin = ( VI_MAX_HEIGHT_NTSC / 2 - 448 / 2 ) / 2 ;
TV_448i . viYOrigin = ( VI_MAX_HEIGHT_NTSC - 448 ) / 2 ;
break ;
2008-08-23 23:15:58 +00:00
2008-09-11 04:41:58 +00:00
default :
2008-09-11 02:23:48 +00:00
// 480 lines (PAL 60Hz)
2008-08-06 01:39:43 +00:00
vmode_60hz = 1 ;
2008-11-04 19:10:23 +00:00
// Original Video modes (forced to PAL 60hz)
// set video signal mode
TV_239p . viTVMode = VI_TVMODE ( vmode - > viTVMode > > 2 , VI_NON_INTERLACE ) ;
TV_478i . viTVMode = VI_TVMODE ( vmode - > viTVMode > > 2 , VI_INTERLACE ) ;
TV_224p . viTVMode = VI_TVMODE ( vmode - > viTVMode > > 2 , VI_NON_INTERLACE ) ;
TV_448i . viTVMode = VI_TVMODE ( vmode - > viTVMode > > 2 , VI_INTERLACE ) ;
// set VI position
TV_239p . viYOrigin = ( VI_MAX_HEIGHT_NTSC / 2 - 478 / 2 ) / 2 ;
TV_478i . viYOrigin = ( VI_MAX_HEIGHT_NTSC - 478 ) / 2 ;
TV_224p . viYOrigin = ( VI_MAX_HEIGHT_NTSC / 2 - 448 / 2 ) / 2 ;
TV_448i . viYOrigin = ( VI_MAX_HEIGHT_NTSC - 448 ) / 2 ;
2008-08-06 01:39:43 +00:00
break ;
}
2008-09-10 17:33:35 +00:00
2008-11-04 20:43:15 +00:00
# ifdef HW_DOL
/* we have component cables, but the preferred mode is interlaced
* why don ' t we switch into progressive ?
* on the Wii , the user can do this themselves on their Wii Settings */
2008-10-31 07:32:35 +00:00
if ( VIDEO_HaveComponentCable ( ) )
2008-09-30 07:22:40 +00:00
vmode = & TVNtsc480Prog ;
2008-11-04 20:43:15 +00:00
# endif
2008-09-30 07:22:40 +00:00
2008-09-09 03:02:11 +00:00
// check for progressive scan
2008-09-11 02:23:48 +00:00
if ( vmode - > viTVMode = = VI_TVMODE_NTSC_PROG )
progressive = true ;
2008-09-11 04:41:58 +00:00
2008-11-06 06:34:18 +00:00
# ifdef HW_RVL
2008-11-05 08:37:58 +00:00
// widescreen fix
2008-11-05 17:26:59 +00:00
if ( CONF_GetAspectRatio ( ) )
{
vmode - > viWidth = 678 ;
vmode - > viXOrigin = ( VI_MAX_WIDTH_PAL - 678 ) / 2 ;
}
2008-11-06 06:34:18 +00:00
# endif
2008-11-05 08:37:58 +00:00
2009-01-02 22:32:16 +00:00
VIDEO_Configure ( vmode ) ;
2008-09-10 17:33:35 +00:00
2009-01-02 22:32:16 +00:00
screenheight = vmode - > xfbHeight ;
2008-08-10 03:14:39 +00:00
2008-08-23 23:15:58 +00:00
// Allocate the video buffers
2009-01-02 22:32:16 +00:00
xfb [ 0 ] = ( u32 * ) MEM_K0_TO_K1 ( SYS_AllocateFramebuffer ( vmode ) ) ;
xfb [ 1 ] = ( u32 * ) MEM_K0_TO_K1 ( SYS_AllocateFramebuffer ( vmode ) ) ;
2008-08-10 03:14:39 +00:00
2009-01-02 22:32:16 +00:00
// A console is always useful while debugging
console_init ( xfb [ 0 ] , 20 , 64 , vmode - > fbWidth , vmode - > xfbHeight , vmode - > fbWidth * 2 ) ;
2008-08-10 03:14:39 +00:00
2009-01-02 22:32:16 +00:00
// Clear framebuffers etc.
VIDEO_ClearFrameBuffer ( vmode , xfb [ 0 ] , COLOR_BLACK ) ;
VIDEO_ClearFrameBuffer ( vmode , xfb [ 1 ] , COLOR_BLACK ) ;
VIDEO_SetNextFramebuffer ( xfb [ 0 ] ) ;
2008-08-10 03:14:39 +00:00
2008-08-23 23:15:58 +00:00
// video callbacks
2008-08-06 01:39:43 +00:00
VIDEO_SetPostRetraceCallback ( ( VIRetraceCallback ) UpdatePadsCB ) ;
2009-01-02 22:32:16 +00:00
VIDEO_SetPreRetraceCallback ( ( VIRetraceCallback ) copy_to_xfb ) ;
2008-09-10 17:33:35 +00:00
2009-01-02 22:32:16 +00:00
VIDEO_SetBlack ( FALSE ) ;
VIDEO_Flush ( ) ;
VIDEO_WaitVSync ( ) ;
if ( vmode - > viTVMode & VI_NON_INTERLACE )
2008-08-23 23:15:58 +00:00
VIDEO_WaitVSync ( ) ;
2008-08-10 03:14:39 +00:00
2009-01-02 22:32:16 +00:00
copynow = GX_FALSE ;
StartGX ( ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
draw_init ( ) ;
2008-08-10 03:14:39 +00:00
2009-01-02 22:32:16 +00:00
InitVideoThread ( ) ;
2008-08-10 03:14:39 +00:00
2009-01-02 22:32:16 +00:00
// Finally, the video is up and ready for use :)
2008-08-06 02:31:44 +00:00
}
2008-08-06 01:09:59 +00:00
/****************************************************************************
2008-09-12 05:28:40 +00:00
* ResetVideo_Emu
*
* Reset the video / rendering mode for the emulator rendering
2008-08-23 23:15:58 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
void
2008-08-23 23:15:58 +00:00
ResetVideo_Emu ( )
2008-08-06 01:09:59 +00:00
{
2008-08-23 23:15:58 +00:00
GXRModeObj * rmode ;
2008-12-30 07:30:31 +00:00
Mtx44 p ;
2008-09-10 17:33:35 +00:00
2008-09-23 03:49:57 +00:00
int i = - 1 ;
2008-09-11 02:23:48 +00:00
if ( GCSettings . render = = 0 ) // original render mode
2008-08-23 23:15:58 +00:00
{
2008-11-04 20:43:15 +00:00
for ( i = 0 ; i < 4 ; i + + )
{
if ( tvmodes [ i ] - > efbHeight = = vheight )
2008-09-09 03:02:11 +00:00
break ;
2008-08-23 23:15:58 +00:00
}
rmode = tvmodes [ i ] ;
2008-09-10 17:33:35 +00:00
}
2008-11-04 19:10:23 +00:00
else
2008-09-11 02:23:48 +00:00
{
2008-08-23 23:15:58 +00:00
rmode = vmode ; // same mode as menu
2008-09-11 02:23:48 +00:00
}
2008-08-23 23:15:58 +00:00
VIDEO_Configure ( rmode ) ;
VIDEO_Flush ( ) ;
2008-11-04 19:10:23 +00:00
VIDEO_WaitVSync ( ) ;
2008-11-04 20:43:15 +00:00
if ( rmode - > viTVMode & VI_NON_INTERLACE )
VIDEO_WaitVSync ( ) ;
else
while ( VIDEO_GetNextField ( ) )
VIDEO_WaitVSync ( ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetViewport ( 0 , 0 , rmode - > fbWidth , rmode - > efbHeight , 0 , 1 ) ;
GX_SetDispCopyYScale ( ( f32 ) rmode - > xfbHeight / ( f32 ) rmode - > efbHeight ) ;
GX_SetScissor ( 0 , 0 , rmode - > fbWidth , rmode - > efbHeight ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetDispCopySrc ( 0 , 0 , rmode - > fbWidth , rmode - > efbHeight ) ;
GX_SetDispCopyDst ( rmode - > fbWidth , rmode - > xfbHeight ) ;
2008-11-04 19:10:23 +00:00
GX_SetCopyFilter ( rmode - > aa , rmode - > sample_pattern , ( GCSettings . render = = 1 ) ? GX_TRUE : GX_FALSE , rmode - > vfilter ) ; // deflicker ON only for filtered mode
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetFieldMode ( rmode - > field_rendering , ( ( rmode - > viHeight = = 2 * rmode - > xfbHeight ) ? GX_ENABLE : GX_DISABLE ) ) ;
GX_SetPixelFmt ( GX_PF_RGB8_Z24 , GX_ZC_LINEAR ) ;
2008-11-04 19:10:23 +00:00
guOrtho ( p , rmode - > efbHeight / 2 , - ( rmode - > efbHeight / 2 ) , - ( rmode - > fbWidth / 2 ) , rmode - > fbWidth / 2 , 100 , 1000 ) ; // matrix, t, b, l, r, n, f
2008-08-23 23:15:58 +00:00
GX_LoadProjectionMtx ( p , GX_ORTHOGRAPHIC ) ;
2008-09-10 17:33:35 +00:00
2008-09-23 03:49:57 +00:00
# ifdef _DEBUG_VIDEO
// log stuff
fprintf ( debughandle , " \n \n rmode = tvmodes[%d], field_rendering: %d " , i , ( rmode - > viHeight = = 2 * rmode - > xfbHeight ) ) ;
fprintf ( debughandle , " \n Interlaced: %i, \t vwidth: %d, vheight: %d, \t fb_W: %u, efb_H: %u " , IPPU . Interlace , vwidth , vheight , rmode - > fbWidth , rmode - > efbHeight ) ;
# endif
2008-08-06 01:09:59 +00:00
}
/****************************************************************************
2008-09-12 05:28:40 +00:00
* ResetVideo_Menu
*
* Reset the video / rendering mode for the menu
2008-08-23 23:15:58 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
void
2008-08-23 23:15:58 +00:00
ResetVideo_Menu ( )
2008-08-06 01:09:59 +00:00
{
2008-12-30 07:30:31 +00:00
Mtx44 p ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
VIDEO_Configure ( vmode ) ;
VIDEO_ClearFrameBuffer ( vmode , xfb [ whichfb ] , COLOR_BLACK ) ;
VIDEO_Flush ( ) ;
VIDEO_WaitVSync ( ) ;
2008-11-04 20:43:15 +00:00
if ( vmode - > viTVMode & VI_NON_INTERLACE )
VIDEO_WaitVSync ( ) ;
else
while ( VIDEO_GetNextField ( ) )
VIDEO_WaitVSync ( ) ;
2008-08-23 23:15:58 +00:00
GX_SetViewport ( 0 , 0 , vmode - > fbWidth , vmode - > efbHeight , 0 , 1 ) ;
GX_SetDispCopyYScale ( ( f32 ) vmode - > xfbHeight / ( f32 ) vmode - > efbHeight ) ;
GX_SetScissor ( 0 , 0 , vmode - > fbWidth , vmode - > efbHeight ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetDispCopySrc ( 0 , 0 , vmode - > fbWidth , vmode - > efbHeight ) ;
GX_SetDispCopyDst ( vmode - > fbWidth , vmode - > xfbHeight ) ;
GX_SetCopyFilter ( vmode - > aa , vmode - > sample_pattern , GX_TRUE , vmode - > vfilter ) ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
GX_SetFieldMode ( vmode - > field_rendering , ( ( vmode - > viHeight = = 2 * vmode - > xfbHeight ) ? GX_ENABLE : GX_DISABLE ) ) ;
GX_SetPixelFmt ( GX_PF_RGB8_Z24 , GX_ZC_LINEAR ) ;
guOrtho ( p , vmode - > efbHeight / 2 , - ( vmode - > efbHeight / 2 ) , - ( vmode - > fbWidth / 2 ) , vmode - > fbWidth / 2 , 10 , 1000 ) ; // matrix, t, b, l, r, n, f
GX_LoadProjectionMtx ( p , GX_ORTHOGRAPHIC ) ;
2008-08-06 01:09:59 +00:00
}
/****************************************************************************
* Update Video
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-25 05:57:43 +00:00
uint32 prevRenderedFrameCount = 0 ;
extern bool CheckVideo ;
2008-08-06 01:09:59 +00:00
void
update_video ( int width , int height )
{
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
vwidth = width ;
vheight = height ;
2008-08-06 01:09:59 +00:00
2008-09-11 02:23:48 +00:00
// Ensure previous vb has complete
2008-08-23 23:15:58 +00:00
while ( ( LWP_ThreadIsSuspended ( vbthread ) = = 0 ) | | ( copynow = = GX_TRUE ) )
{
2008-11-04 20:43:15 +00:00
usleep ( 50 ) ;
2008-08-23 23:15:58 +00:00
}
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
whichfb ^ = 1 ;
2008-09-09 21:26:12 +00:00
if ( oldvheight ! = vheight | | oldvwidth ! = vwidth ) // if rendered width/height changes, update scaling
CheckVideo = 1 ;
2008-09-10 17:33:35 +00:00
2008-09-09 21:26:12 +00:00
if ( CheckVideo & & ( IPPU . RenderedFramesCount ! = prevRenderedFrameCount ) ) // if we get back from the menu, and have rendered at least 1 frame
2008-08-23 23:15:58 +00:00
{
2008-09-28 02:13:19 +00:00
int xscale , yscale ;
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
ResetVideo_Emu ( ) ; // reset video to emulator rendering settings
2008-09-10 17:33:35 +00:00
2008-08-06 01:09:59 +00:00
/** Update scaling **/
2008-09-11 02:23:48 +00:00
if ( GCSettings . render = = 0 ) // original render mode
2008-08-23 23:15:58 +00:00
{
2008-11-04 19:10:23 +00:00
xscale = 256 ;
2008-08-23 23:15:58 +00:00
yscale = vheight / 2 ;
2008-11-04 20:43:15 +00:00
}
else // unfiltered and filtered mode
{
2008-11-04 19:10:23 +00:00
xscale = 320 ;
2008-11-07 18:23:26 +00:00
yscale = ( vheight > ( vmode - > efbHeight / 2 ) ) ? ( vheight / 2 ) : vheight ;
2008-08-23 23:15:58 +00:00
}
2008-09-10 17:33:35 +00:00
2008-08-25 05:57:43 +00:00
// aspect ratio scaling (change width scale)
// yes its pretty cheap and ugly, but its easy!
2008-09-11 02:23:48 +00:00
if ( GCSettings . widescreen )
2008-10-22 18:52:51 +00:00
xscale = ( 3 * xscale ) / 4 ;
2008-09-27 07:13:52 +00:00
2008-10-24 05:11:01 +00:00
xscale * = GCSettings . ZoomLevel ;
yscale * = GCSettings . ZoomLevel ;
2008-09-10 17:33:35 +00:00
2008-09-28 02:13:19 +00:00
square [ 6 ] = square [ 3 ] = xscale + GCSettings . xshift ;
square [ 0 ] = square [ 9 ] = - xscale + GCSettings . xshift ;
2008-09-30 06:43:34 +00:00
square [ 4 ] = square [ 1 ] = yscale - GCSettings . yshift ;
square [ 7 ] = square [ 10 ] = - yscale - GCSettings . yshift ;
2008-11-07 16:50:42 +00:00
DCFlushRange ( square , 32 ) ; // update memory BEFORE the GPU accesses it!
2008-11-04 19:10:23 +00:00
draw_init ( ) ;
2008-08-23 23:15:58 +00:00
2008-09-11 02:23:48 +00:00
GX_InitTexObj ( & texobj , texturemem , vwidth , vheight , GX_TF_RGB565 , GX_CLAMP , GX_CLAMP , GX_FALSE ) ; // initialize the texture obj we are going to use
2008-09-10 17:33:35 +00:00
2008-09-11 02:23:48 +00:00
if ( GCSettings . render = = 0 | | GCSettings . render = = 2 )
GX_InitTexObjLOD ( & texobj , GX_NEAR , GX_NEAR_MIP_NEAR , 2.5 , 9.0 , 0.0 , GX_FALSE , GX_FALSE , GX_ANISO_1 ) ; // original/unfiltered video mode: force texture filtering OFF
2008-09-11 04:41:58 +00:00
2008-09-11 02:23:48 +00:00
GX_LoadTexObj ( & texobj , GX_TEXMAP0 ) ; // load texture object so its ready to use
2008-09-10 17:33:35 +00:00
2008-09-23 03:49:57 +00:00
# ifdef _DEBUG_VIDEO
// log stuff
fprintf ( debughandle , " \n xscale: %d, yscale: %d " , xscale , yscale ) ;
# endif
2008-09-10 17:33:35 +00:00
2008-08-23 23:15:58 +00:00
oldvwidth = vwidth ;
oldvheight = vheight ;
2008-08-25 05:57:43 +00:00
CheckVideo = 0 ;
2008-08-23 23:15:58 +00:00
}
2008-09-11 04:41:58 +00:00
2009-01-16 04:48:15 +00:00
MakeTexture565 ( ( char * ) GFX . Screen , ( char * ) texturemem , vwidth , vheight ) ; // convert image to texture
2008-08-06 01:09:59 +00:00
2008-09-11 02:23:48 +00:00
DCFlushRange ( texturemem , TEX_WIDTH * TEX_HEIGHT * 2 ) ; // update the texture memory
GX_InvalidateTexAll ( ) ;
2008-08-06 01:09:59 +00:00
2008-09-11 02:23:48 +00:00
draw_square ( view ) ; // draw the quad
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
GX_DrawDone ( ) ;
VIDEO_SetNextFramebuffer ( xfb [ whichfb ] ) ;
VIDEO_Flush ( ) ;
copynow = GX_TRUE ;
2008-08-06 02:31:44 +00:00
2008-09-11 02:23:48 +00:00
// Return to caller, don't waste time waiting for vb
2008-08-23 23:15:58 +00:00
LWP_ResumeThread ( vbthread ) ;
}
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
/****************************************************************************
* Zoom Functions
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-23 23:15:58 +00:00
void
zoom ( float speed )
{
2008-10-24 05:11:01 +00:00
if ( GCSettings . ZoomLevel > 1 )
GCSettings . ZoomLevel + = ( speed / - 100.0 ) ;
2008-09-23 03:49:57 +00:00
else
2008-10-24 05:11:01 +00:00
GCSettings . ZoomLevel + = ( speed / - 200.0 ) ;
2008-09-27 07:13:52 +00:00
2008-10-24 05:11:01 +00:00
if ( GCSettings . ZoomLevel < 0.5 )
GCSettings . ZoomLevel = 0.5 ;
2008-10-29 05:48:40 +00:00
else if ( GCSettings . ZoomLevel > 2.0 )
GCSettings . ZoomLevel = 2.0 ;
2008-09-27 07:13:52 +00:00
2008-08-23 23:15:58 +00:00
oldvheight = 0 ; // update video
}
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
void
zoom_reset ( )
{
2008-10-24 05:11:01 +00:00
GCSettings . ZoomLevel = 1.0 ;
2008-08-23 23:15:58 +00:00
oldvheight = 0 ; // update video
2008-08-06 01:09:59 +00:00
}
2008-08-23 23:15:58 +00:00
/****************************************************************************
* Drawing screen
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-06 01:09:59 +00:00
void
2008-08-23 23:15:58 +00:00
clearscreen ( int colour )
2008-08-06 01:09:59 +00:00
{
2008-08-23 23:15:58 +00:00
whichfb ^ = 1 ;
VIDEO_ClearFrameBuffer ( vmode , xfb [ whichfb ] , colour ) ;
# ifdef HW_RVL
// on wii copy from memory
memcpy ( ( char * ) xfb [ whichfb ] , ( char * ) backdrop , 640 * screenheight * 2 ) ;
# else
// on gc copy from aram
ARAMFetch ( ( char * ) xfb [ whichfb ] , ( char * ) AR_BACKDROP , 640 * screenheight * 2 ) ;
# endif
}
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
void
showscreen ( )
{
copynow = GX_FALSE ;
VIDEO_SetNextFramebuffer ( xfb [ whichfb ] ) ;
VIDEO_Flush ( ) ;
VIDEO_WaitVSync ( ) ;
}
2008-08-06 01:09:59 +00:00
2008-08-23 23:15:58 +00:00
/****************************************************************************
* setGFX
*
* Setup the global GFX information for Snes9x
2008-09-12 05:28:40 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-23 23:15:58 +00:00
void
setGFX ( )
{
GFX . Screen = ( unsigned short * ) snes9xgfx ;
GFX . Pitch = 1024 ;
2008-08-06 01:09:59 +00:00
}