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-July 2007
|
2008-09-23 04:13:33 +00:00
|
|
|
* Michniewski 2008
|
2008-09-12 05:28:40 +00:00
|
|
|
* Tantric September 2008
|
2008-08-06 01:09:59 +00:00
|
|
|
*
|
|
|
|
* snes9xGX.cpp
|
|
|
|
*
|
|
|
|
* This file controls overall program flow. Most things start and end here!
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
|
|
|
|
#include <gccore.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2008-08-06 01:39:43 +00:00
|
|
|
#include <ogcsys.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <wiiuse/wpad.h>
|
|
|
|
#include <sdcard/card_cmn.h>
|
|
|
|
#include <sdcard/wiisd_io.h>
|
|
|
|
#include <sdcard/card_io.h>
|
2008-08-07 03:25:02 +00:00
|
|
|
#include <fat.h>
|
2008-08-06 01:39:43 +00:00
|
|
|
|
2008-08-19 06:05:57 +00:00
|
|
|
#ifdef WII_DVD
|
2008-08-23 03:20:54 +00:00
|
|
|
extern "C" {
|
2008-08-19 06:05:57 +00:00
|
|
|
#include <di/di.h>
|
2008-08-23 03:20:54 +00:00
|
|
|
}
|
|
|
|
#endif
|
2008-08-19 06:05:57 +00:00
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "snes9x.h"
|
|
|
|
#include "memmap.h"
|
|
|
|
#include "debug.h"
|
|
|
|
#include "cpuexec.h"
|
|
|
|
#include "ppu.h"
|
|
|
|
#include "apu.h"
|
|
|
|
#include "display.h"
|
|
|
|
#include "gfx.h"
|
|
|
|
#include "soundux.h"
|
|
|
|
#include "spc700.h"
|
|
|
|
#include "spc7110.h"
|
|
|
|
#include "controls.h"
|
|
|
|
|
|
|
|
#include "snes9xGX.h"
|
2008-09-27 07:13:52 +00:00
|
|
|
#include "aram.h"
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "dvd.h"
|
2008-08-10 03:14:39 +00:00
|
|
|
#include "smbop.h"
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "video.h"
|
2008-08-07 05:19:17 +00:00
|
|
|
#include "menudraw.h"
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "s9xconfig.h"
|
|
|
|
#include "audio.h"
|
|
|
|
#include "menu.h"
|
|
|
|
#include "sram.h"
|
2008-08-12 03:25:16 +00:00
|
|
|
#include "freeze.h"
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "preferences.h"
|
2008-08-06 01:45:56 +00:00
|
|
|
#include "button_mapping.h"
|
2008-08-07 05:19:17 +00:00
|
|
|
#include "fileop.h"
|
2008-08-20 04:07:38 +00:00
|
|
|
#include "input.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
|
|
|
int ConfigRequested = 0;
|
2008-09-23 03:49:57 +00:00
|
|
|
FILE* debughandle;
|
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
extern int FrameTimer;
|
|
|
|
|
2008-08-06 01:39:43 +00:00
|
|
|
extern long long prev;
|
2008-08-06 01:09:59 +00:00
|
|
|
extern unsigned int timediffallowed;
|
|
|
|
|
2008-10-21 08:08:30 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* ipl_set_config
|
|
|
|
* lowlevel Qoob Modchip disable
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
void ipl_set_config(unsigned char c)
|
|
|
|
{
|
|
|
|
volatile unsigned long* exi = (volatile unsigned long*)0xCC006800;
|
|
|
|
unsigned long val,addr;
|
|
|
|
addr=0xc0000000;
|
|
|
|
val = c << 24;
|
|
|
|
exi[0] = ((((exi[0]) & 0x405) | 256) | 48); //select IPL
|
|
|
|
//write addr of IPL
|
|
|
|
exi[0 * 5 + 4] = addr;
|
|
|
|
exi[0 * 5 + 3] = ((4 - 1) << 4) | (1 << 2) | 1;
|
|
|
|
while (exi[0 * 5 + 3] & 1);
|
|
|
|
//write the ipl we want to send
|
|
|
|
exi[0 * 5 + 4] = val;
|
|
|
|
exi[0 * 5 + 3] = ((4 - 1) << 4) | (1 << 2) | 1;
|
|
|
|
while (exi[0 * 5 + 3] & 1);
|
|
|
|
|
|
|
|
exi[0] &= 0x405; //deselect IPL
|
|
|
|
}
|
2008-08-06 01:45:56 +00:00
|
|
|
|
2008-08-06 01:55:59 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* setFrameTimerMethod()
|
2008-08-07 05:19:17 +00:00
|
|
|
* change frametimer method depending on whether ROM is NTSC or PAL
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:55:59 +00:00
|
|
|
extern u8 vmode_60hz;
|
|
|
|
int timerstyle;
|
2008-08-06 01:09:59 +00:00
|
|
|
|
2008-08-06 01:55:59 +00:00
|
|
|
void setFrameTimerMethod()
|
|
|
|
{
|
|
|
|
/*
|
2008-08-07 05:19:17 +00:00
|
|
|
Set frametimer method
|
|
|
|
(timerstyle: 0=NTSC vblank, 1=PAL int timer)
|
2008-08-06 01:55:59 +00:00
|
|
|
*/
|
|
|
|
if ( Settings.PAL ) {
|
|
|
|
if(vmode_60hz == 1)
|
|
|
|
timerstyle = 1;
|
|
|
|
else
|
|
|
|
timerstyle = 0;
|
|
|
|
//timediffallowed = Settings.FrameTimePAL;
|
|
|
|
} else {
|
|
|
|
if(vmode_60hz == 1)
|
|
|
|
timerstyle = 0;
|
|
|
|
else
|
|
|
|
timerstyle = 1;
|
|
|
|
//timediffallowed = Settings.FrameTimeNTSC;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2008-08-06 01:09:59 +00:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Emulation loop
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
extern void S9xInitSync();
|
2008-09-12 05:28:40 +00:00
|
|
|
bool CheckVideo = 0; // for forcing video reset in video.cpp
|
2008-08-25 05:57:43 +00:00
|
|
|
extern uint32 prevRenderedFrameCount;
|
2008-08-06 01:09:59 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
emulate ()
|
|
|
|
{
|
2008-08-06 01:39:43 +00:00
|
|
|
S9xSetSoundMute (TRUE);
|
|
|
|
AudioStart ();
|
2008-09-12 05:28:40 +00:00
|
|
|
S9xInitSync(); // Eke-Eke: initialize frame Sync
|
2008-08-06 01:55:59 +00:00
|
|
|
|
2008-09-12 05:28:40 +00:00
|
|
|
setFrameTimerMethod(); // set frametimer method every time a ROM is loaded
|
2008-08-06 01:09:59 +00:00
|
|
|
|
2008-08-06 01:39:43 +00:00
|
|
|
while (1)
|
|
|
|
{
|
2008-08-20 04:07:38 +00:00
|
|
|
S9xMainLoop ();
|
|
|
|
NGCReportButtons ();
|
2008-08-20 07:58:55 +00:00
|
|
|
|
2008-08-20 04:07:38 +00:00
|
|
|
if (ConfigRequested)
|
|
|
|
{
|
2008-08-23 23:15:58 +00:00
|
|
|
// change to menu video mode
|
|
|
|
ResetVideo_Menu ();
|
2008-08-20 04:07:38 +00:00
|
|
|
|
|
|
|
if ( GCSettings.AutoSave == 1 )
|
|
|
|
{
|
|
|
|
SaveSRAM ( GCSettings.SaveMethod, SILENT );
|
|
|
|
}
|
|
|
|
else if ( GCSettings.AutoSave == 2 )
|
|
|
|
{
|
|
|
|
if ( WaitPromptChoice ((char*)"Save Freeze State?", (char*)"Don't Save", (char*)"Save") )
|
|
|
|
NGCFreezeGame ( GCSettings.SaveMethod, SILENT );
|
|
|
|
}
|
|
|
|
else if ( GCSettings.AutoSave == 3 )
|
|
|
|
{
|
|
|
|
if ( WaitPromptChoice ((char*)"Save SRAM and Freeze State?", (char*)"Don't Save", (char*)"Save") )
|
|
|
|
{
|
|
|
|
SaveSRAM(GCSettings.SaveMethod, SILENT );
|
|
|
|
NGCFreezeGame ( GCSettings.SaveMethod, SILENT );
|
|
|
|
}
|
|
|
|
}
|
2008-09-10 05:57:37 +00:00
|
|
|
|
2008-09-09 03:02:11 +00:00
|
|
|
// GUI Stuff
|
2008-09-09 16:24:42 +00:00
|
|
|
/*
|
2008-09-09 21:26:12 +00:00
|
|
|
gui_alloc();
|
2008-09-09 03:02:11 +00:00
|
|
|
gui_makebg ();
|
|
|
|
gui_clearscreen();
|
|
|
|
gui_draw ();
|
|
|
|
gui_showscreen ();
|
|
|
|
//gui_savescreen ();
|
2008-09-09 16:24:42 +00:00
|
|
|
*/
|
2008-08-20 04:07:38 +00:00
|
|
|
|
2008-09-30 05:31:46 +00:00
|
|
|
MainMenu (2); // go to game menu
|
2008-08-20 07:58:55 +00:00
|
|
|
|
2008-10-28 05:13:24 +00:00
|
|
|
// save zoom level
|
|
|
|
SavePrefs(GCSettings.SaveMethod, SILENT);
|
|
|
|
|
2008-08-20 04:07:38 +00:00
|
|
|
FrameTimer = 0;
|
2008-08-23 23:15:58 +00:00
|
|
|
setFrameTimerMethod (); // set frametimer method every time a ROM is loaded
|
2008-08-20 07:58:55 +00:00
|
|
|
|
2008-08-20 04:07:38 +00:00
|
|
|
Settings.SuperScopeMaster = (GCSettings.Superscope > 0 ? true : false);
|
|
|
|
Settings.MouseMaster = (GCSettings.Mouse > 0 ? true : false);
|
|
|
|
Settings.JustifierMaster = (GCSettings.Justifier > 0 ? true : false);
|
2008-08-23 23:15:58 +00:00
|
|
|
SetControllers ();
|
|
|
|
//S9xReportControllers ();
|
2008-08-20 07:58:55 +00:00
|
|
|
|
2008-08-20 04:07:38 +00:00
|
|
|
ConfigRequested = 0;
|
2008-09-27 07:13:52 +00:00
|
|
|
|
2008-09-23 03:49:57 +00:00
|
|
|
#ifdef _DEBUG_VIDEO
|
|
|
|
// log stuff
|
|
|
|
fprintf(debughandle, "\n\nPlaying ROM: %s", Memory.ROMFilename);
|
|
|
|
fprintf(debughandle, "\nrender: %u", GCSettings.render);
|
|
|
|
#endif
|
2008-09-04 02:42:27 +00:00
|
|
|
|
2008-08-25 05:57:43 +00:00
|
|
|
CheckVideo = 1; // force video update
|
|
|
|
prevRenderedFrameCount = IPPU.RenderedFramesCount;
|
2008-09-04 02:42:27 +00:00
|
|
|
|
2008-09-23 03:49:57 +00:00
|
|
|
|
2008-08-20 04:07:38 +00:00
|
|
|
}//if ConfigRequested
|
2008-08-20 07:58:55 +00:00
|
|
|
|
2008-08-20 04:07:38 +00:00
|
|
|
}//while
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* MAIN
|
|
|
|
*
|
|
|
|
* Steps to Snes9x Emulation :
|
|
|
|
* 1. Initialise GC Video
|
|
|
|
* 2. Initialise libfreetype (Nice to read something)
|
|
|
|
* 3. Set S9xSettings to standard defaults (s9xconfig.h)
|
|
|
|
* 4. Allocate Snes9x Memory
|
|
|
|
* 5. Allocate APU Memory
|
|
|
|
* 6. Set Pixel format to RGB565 for GL Rendering
|
|
|
|
* 7. Initialise Snes9x/GC Sound System
|
|
|
|
* 8. Initialise Snes9x Graphics subsystem
|
|
|
|
* 9. Let's Party!
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
int
|
|
|
|
main ()
|
|
|
|
{
|
2008-10-21 08:08:30 +00:00
|
|
|
#ifdef HW_DOL
|
|
|
|
ipl_set_config(6); // disable Qoob modchip
|
|
|
|
#endif
|
|
|
|
|
2008-08-19 06:05:57 +00:00
|
|
|
#ifdef WII_DVD
|
|
|
|
DI_Init(); // first
|
|
|
|
#endif
|
2008-08-20 07:58:55 +00:00
|
|
|
|
2008-08-07 03:25:02 +00:00
|
|
|
int selectedMenu = -1;
|
2008-08-20 07:58:55 +00:00
|
|
|
|
2008-09-27 07:13:52 +00:00
|
|
|
// Initialise video
|
|
|
|
InitGCVideo ();
|
|
|
|
|
|
|
|
// Controllers
|
2008-08-19 06:05:57 +00:00
|
|
|
#ifdef HW_RVL
|
2008-08-06 01:39:43 +00:00
|
|
|
WPAD_Init();
|
2008-08-06 14:40:28 +00:00
|
|
|
// read wiimote accelerometer and IR data
|
|
|
|
WPAD_SetDataFormat(WPAD_CHAN_ALL,WPAD_FMT_BTNS_ACC_IR);
|
|
|
|
WPAD_SetVRes(WPAD_CHAN_ALL,640,480);
|
2008-08-19 06:05:57 +00:00
|
|
|
#endif
|
2008-08-06 01:39:43 +00:00
|
|
|
|
2008-09-27 07:13:52 +00:00
|
|
|
PAD_Init ();
|
|
|
|
|
|
|
|
// Audio
|
|
|
|
AUDIO_Init (NULL);
|
|
|
|
|
|
|
|
// GC Audio RAM (for ROM and backdrop storage)
|
|
|
|
AR_Init (NULL, 0);
|
|
|
|
|
2008-10-09 21:44:53 +00:00
|
|
|
// GameCube only - Injected ROM
|
2008-09-27 07:13:52 +00:00
|
|
|
// Before going any further, let's copy any injected ROM image
|
2008-10-09 21:44:53 +00:00
|
|
|
// We'll put it in ARAM for safe storage
|
|
|
|
|
|
|
|
#ifdef HW_DOL
|
2008-09-27 07:13:52 +00:00
|
|
|
int *romptr = (int *) 0x81000000; // location of injected rom
|
|
|
|
|
|
|
|
if (memcmp ((char *) romptr, "SNESROM0", 8) == 0)
|
|
|
|
{
|
2008-10-09 22:36:22 +00:00
|
|
|
SNESROMSize = romptr[2];
|
2008-10-09 21:44:53 +00:00
|
|
|
|
2008-10-09 22:36:22 +00:00
|
|
|
if(SNESROMSize > (1024*128) && SNESROMSize < (1024*1024*8))
|
2008-10-09 21:44:53 +00:00
|
|
|
{
|
|
|
|
romptr = (int *) 0x81000020;
|
2008-10-09 22:36:22 +00:00
|
|
|
ARAMPut ((char *) romptr, (char *) AR_SNESROM, SNESROMSize);
|
2008-10-09 21:44:53 +00:00
|
|
|
}
|
|
|
|
else // not a valid ROM size
|
|
|
|
{
|
2008-10-09 22:36:22 +00:00
|
|
|
SNESROMSize = 0;
|
2008-10-09 21:44:53 +00:00
|
|
|
}
|
2008-09-27 07:13:52 +00:00
|
|
|
}
|
2008-10-09 21:44:53 +00:00
|
|
|
#endif
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-07 23:50:57 +00:00
|
|
|
// Initialise freetype font system
|
2008-08-06 01:09:59 +00:00
|
|
|
if (FT_Init ())
|
|
|
|
{
|
|
|
|
printf ("Cannot initialise font subsystem!\n");
|
|
|
|
while (1);
|
|
|
|
}
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
unpackbackdrop ();
|
|
|
|
|
2008-09-07 23:50:57 +00:00
|
|
|
// Set defaults
|
2008-08-06 01:09:59 +00:00
|
|
|
DefaultSettings ();
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
S9xUnmapAllControls ();
|
|
|
|
SetDefaultButtonMap ();
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-07 23:50:57 +00:00
|
|
|
// Allocate SNES Memory
|
2008-08-06 01:09:59 +00:00
|
|
|
if (!Memory.Init ())
|
|
|
|
while (1);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-07 23:50:57 +00:00
|
|
|
// Allocate APU
|
2008-08-06 01:09:59 +00:00
|
|
|
if (!S9xInitAPU ())
|
|
|
|
while (1);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-07 23:50:57 +00:00
|
|
|
// Set Pixel Renderer to match 565
|
2008-08-06 01:09:59 +00:00
|
|
|
S9xSetRenderPixelFormat (RGB565);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-07 23:50:57 +00:00
|
|
|
// Initialise Snes Sound System
|
2008-08-06 01:09:59 +00:00
|
|
|
S9xInitSound (5, TRUE, 1024);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-07 23:50:57 +00:00
|
|
|
// Initialise Graphics
|
2008-08-06 01:09:59 +00:00
|
|
|
setGFX ();
|
|
|
|
if (!S9xGraphicsInit ())
|
|
|
|
while (1);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
// Initialize libFAT for SD and USB
|
2008-10-03 06:57:56 +00:00
|
|
|
fatInit (8, false);
|
2008-09-27 07:13:52 +00:00
|
|
|
|
2008-09-23 03:49:57 +00:00
|
|
|
#ifdef _DEBUG_VIDEO
|
|
|
|
// log stuff
|
|
|
|
debughandle = fopen ("log.txt", "wb");
|
|
|
|
#endif
|
2008-08-10 03:14:39 +00:00
|
|
|
|
2008-08-16 00:02:08 +00:00
|
|
|
// Initialize DVD subsystem (GameCube only)
|
|
|
|
#ifndef HW_RVL
|
2008-08-10 03:14:39 +00:00
|
|
|
DVD_Init ();
|
2008-08-16 00:02:08 +00:00
|
|
|
#endif
|
2008-08-10 03:14:39 +00:00
|
|
|
|
2008-09-10 05:57:37 +00:00
|
|
|
// Check if DVD drive belongs to a Wii
|
|
|
|
SetDVDDriveType();
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
// Load preferences
|
2008-09-04 02:42:27 +00:00
|
|
|
if(!LoadPrefs())
|
2008-08-07 03:25:02 +00:00
|
|
|
{
|
|
|
|
WaitPrompt((char*) "Preferences reset - check settings!");
|
2008-09-30 05:31:46 +00:00
|
|
|
selectedMenu = 1; // change to preferences menu
|
2008-08-07 03:25:02 +00:00
|
|
|
}
|
|
|
|
|
2008-10-09 21:44:53 +00:00
|
|
|
// GameCube only - Injected ROM
|
|
|
|
// Everything's been initialized, we can copy our ROM back
|
|
|
|
// from ARAM into main memory
|
|
|
|
|
|
|
|
#ifdef HW_DOL
|
2008-10-09 22:36:22 +00:00
|
|
|
if(SNESROMSize > 0)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
2008-10-09 22:36:22 +00:00
|
|
|
ARAMFetchSlow( (char *)Memory.ROM, (char *)AR_SNESROM, SNESROMSize);
|
2008-10-09 21:44:53 +00:00
|
|
|
Memory.LoadROM ("BLANK.SMC");
|
|
|
|
Memory.LoadSRAM ("BLANK");
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
2008-10-09 21:44:53 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Get the user to load a ROM
|
2008-10-09 22:36:22 +00:00
|
|
|
while (SNESROMSize <= 0)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
2008-10-09 21:44:53 +00:00
|
|
|
MainMenu (selectedMenu);
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-12 05:28:40 +00:00
|
|
|
// Emulate
|
2008-08-06 01:09:59 +00:00
|
|
|
emulate ();
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-09-12 05:28:40 +00:00
|
|
|
// NO! - We're never leaving here!
|
2008-08-06 01:09:59 +00:00
|
|
|
while (1);
|
2008-08-10 03:14:39 +00:00
|
|
|
return 0;
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|