snes9xgx/source/ngc/snes9xGX.cpp

456 lines
9.3 KiB
C++
Raw Normal View History

/****************************************************************************
* Snes9x 1.51 Nintendo Wii/Gamecube Port
*
* softdev July 2006
* crunchy2 May 2007-July 2007
2008-09-23 06:13:33 +02:00
* Michniewski 2008
2009-03-12 08:07:52 +01:00
* Tantric 2008-2009
*
* snes9xGX.cpp
*
* This file controls overall program flow. Most things start and end here!
***************************************************************************/
#include <gccore.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ogcsys.h>
#include <unistd.h>
#include <wiiuse/wpad.h>
#include <fat.h>
#include <debug.h>
2009-09-25 20:50:35 +02:00
#include <sys/iosupport.h>
2009-03-11 18:28:37 +01:00
#ifdef HW_RVL
#include <di/di.h>
2008-08-23 05:20:54 +02:00
#endif
2009-03-11 18:28:37 +01:00
#include "FreeTypeGX.h"
#include "snes9x.h"
#include "memmap.h"
2009-11-25 07:35:14 +01:00
#include "apu/apu.h"
#include "controls.h"
#include "snes9xGX.h"
2008-09-27 09:13:52 +02:00
#include "aram.h"
#include "networkop.h"
#include "video.h"
#include "s9xconfig.h"
#include "audio.h"
#include "menu.h"
#include "sram.h"
2008-08-12 05:25:16 +02:00
#include "freeze.h"
#include "preferences.h"
#include "button_mapping.h"
2008-08-07 07:19:17 +02:00
#include "fileop.h"
2009-03-11 18:28:37 +01:00
#include "filebrowser.h"
#include "input.h"
int ScreenshotRequested = 0;
int ConfigRequested = 0;
int ShutdownRequested = 0;
int ResetRequested = 0;
2009-03-11 18:28:37 +01:00
int ExitRequested = 0;
2009-10-02 00:35:12 +02:00
char appPath[1024] = { 0 };
2009-09-25 20:45:51 +02:00
extern "C" {
extern void __exception_setreload(int t);
}
/****************************************************************************
* Shutdown / Reboot / Exit
***************************************************************************/
2008-12-18 19:36:30 +01:00
void ExitCleanup()
{
2009-03-12 08:07:52 +01:00
ShutdownAudio();
2009-03-11 18:28:37 +01:00
StopGX();
2009-05-06 19:28:06 +02:00
HaltDeviceThread();
2008-12-18 19:36:30 +01:00
UnmountAllFAT();
#ifdef HW_RVL
DI_Close();
#endif
}
#ifdef HW_DOL
#define PSOSDLOADID 0x7c6000a6
int *psoid = (int *) 0x80001800;
void (*PSOReload) () = (void (*)()) 0x80001800;
#endif
2009-03-11 18:28:37 +01:00
void ExitApp()
{
2009-10-06 08:37:53 +02:00
#ifdef HW_RVL
ShutoffRumble();
#endif
SavePrefs(SILENT);
if (SNESROMSize > 0 && !ConfigRequested && GCSettings.AutoSave == 1)
2009-10-02 00:35:12 +02:00
SaveSRAMAuto(SILENT);
2008-12-18 19:36:30 +01:00
ExitCleanup();
if(ShutdownRequested)
SYS_ResetSystem(SYS_POWEROFF, 0, 0);
#ifdef HW_RVL
if(GCSettings.ExitAction == 0) // Auto
2009-03-11 18:28:37 +01:00
{
char * sig = (char *)0x80001804;
if(
sig[0] == 'S' &&
sig[1] == 'T' &&
sig[2] == 'U' &&
sig[3] == 'B' &&
sig[4] == 'H' &&
sig[5] == 'A' &&
sig[6] == 'X' &&
sig[7] == 'X')
GCSettings.ExitAction = 3; // Exit to HBC
else
GCSettings.ExitAction = 1; // HBC not found
2009-03-11 18:28:37 +01:00
}
#endif
if(GCSettings.ExitAction == 1) // Exit to Menu
2009-03-11 18:28:37 +01:00
{
#ifdef HW_RVL
SYS_ResetSystem(SYS_RETURNTOMENU, 0, 0);
#else
#define SOFTRESET_ADR ((volatile u32*)0xCC003024)
*SOFTRESET_ADR = 0x00000000;
#endif
}
else if(GCSettings.ExitAction == 2) // Shutdown Wii
2009-03-11 18:28:37 +01:00
{
SYS_ResetSystem(SYS_POWEROFF, 0, 0);
}
else // Exit to Loader
{
#ifdef HW_RVL
exit(0);
#else
if (psoid[0] == PSOSDLOADID)
PSOReload();
#endif
}
}
#ifdef HW_RVL
void ShutdownCB()
{
ShutdownRequested = 1;
}
void ResetCB()
{
ResetRequested = 1;
}
#endif
#ifdef HW_DOL
2008-10-21 10:08:30 +02: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
}
#endif
/****************************************************************************
* setFrameTimerMethod()
2008-08-07 07:19:17 +02:00
* change frametimer method depending on whether ROM is NTSC or PAL
***************************************************************************/
void setFrameTimerMethod()
{
/*
2008-08-07 07:19:17 +02:00
Set frametimer method
(timerstyle: 0=NTSC vblank, 1=PAL int timer)
*/
if ( Settings.PAL ) {
if(vmode_60hz == 1)
timerstyle = 1;
else
timerstyle = 0;
} else {
if(vmode_60hz == 1)
timerstyle = 0;
else
timerstyle = 1;
}
return;
}
/****************************************************************************
* Emulation loop
***************************************************************************/
extern void S9xInitSync();
extern uint32 prevRenderedFrameCount;
2009-04-02 07:14:18 +02:00
static int videoReset;
static int currentMode;
void
emulate ()
{
2009-03-11 18:28:37 +01:00
while (1) // main loop
{
2009-02-08 23:30:42 +01:00
// go back to checking if devices were inserted/removed
// since we're entering the menu
2009-05-06 19:28:06 +02:00
ResumeDeviceThread();
2008-08-20 09:58:55 +02:00
2009-03-11 18:28:37 +01:00
SwitchAudioMode(1);
2009-03-11 18:28:37 +01:00
if(SNESROMSize == 0)
MainMenu(MENU_GAMESELECTION);
else
MainMenu(MENU_GAME);
2009-03-11 18:28:37 +01:00
AllocGfxMem();
SelectFilterMethod();
2009-02-08 23:30:42 +01:00
ConfigRequested = 0;
ScreenshotRequested = 0;
2009-03-11 18:28:37 +01:00
SwitchAudioMode(0);
2008-09-10 07:57:37 +02:00
2009-03-11 18:28:37 +01:00
Settings.MultiPlayer5Master = (GCSettings.Controller == CTRL_PAD4 ? true : false);
Settings.SuperScopeMaster = (GCSettings.Controller == CTRL_SCOPE ? true : false);
Settings.MouseMaster = (GCSettings.Controller == CTRL_MOUSE ? true : false);
Settings.JustifierMaster = (GCSettings.Controller == CTRL_JUST ? true : false);
2009-02-08 23:30:42 +01:00
SetControllers ();
2009-02-08 23:30:42 +01:00
// stop checking if devices were removed/inserted
// since we're starting emulation again
2009-05-06 19:28:06 +02:00
HaltDeviceThread();
2009-03-11 18:28:37 +01:00
AudioStart ();
2008-08-20 09:58:55 +02:00
2009-02-08 23:30:42 +01:00
FrameTimer = 0;
2009-03-11 18:28:37 +01:00
setFrameTimerMethod (); // set frametimer method every time a ROM is loaded
2008-10-28 06:13:24 +01:00
2009-02-08 23:30:42 +01:00
CheckVideo = 1; // force video update
prevRenderedFrameCount = IPPU.RenderedFramesCount;
2008-08-20 09:58:55 +02:00
2009-04-02 07:14:18 +02:00
videoReset = -1;
currentMode = GCSettings.render;
2009-03-11 18:28:37 +01:00
while(1) // emulation loop
2009-02-08 23:30:42 +01:00
{
S9xMainLoop ();
NGCReportButtons ();
2008-08-20 09:58:55 +02:00
2009-02-08 23:30:42 +01:00
if(ResetRequested)
{
S9xSoftReset (); // reset game
ResetRequested = 0;
}
if (ConfigRequested)
{
2009-06-15 10:23:19 +02:00
ConfigRequested = 0;
FreeGfxMem();
ResetVideo_Menu();
break;
2009-02-08 23:30:42 +01:00
}
#ifdef HW_RVL
if(ShutdownRequested)
ExitApp();
#endif
2009-03-11 18:28:37 +01:00
} // emulation loop
} // main loop
}
2009-09-25 20:45:51 +02:00
/****************************************************************************
* USB Gecko Debugging
***************************************************************************/
static bool gecko = false;
static mutex_t gecko_mutex = 0;
static ssize_t __out_write(struct _reent *r, int fd, const char *ptr, size_t len)
{
u32 level;
if (!ptr || len <= 0 || !gecko)
return -1;
LWP_MutexLock(gecko_mutex);
level = IRQ_Disable();
usb_sendbuffer(1, ptr, len);
IRQ_Restore(level);
LWP_MutexUnlock(gecko_mutex);
return len;
}
const devoptab_t gecko_out = {
"stdout", // device name
0, // size of file structure
NULL, // device open
NULL, // device close
__out_write,// device write
NULL, // device read
NULL, // device seek
NULL, // device fstat
NULL, // device stat
NULL, // device link
NULL, // device unlink
NULL, // device chdir
NULL, // device rename
NULL, // device mkdir
0, // dirStateSize
NULL, // device diropen_r
NULL, // device dirreset_r
NULL, // device dirnext_r
NULL, // device dirclose_r
NULL // device statvfs_r
};
2009-09-25 20:47:39 +02:00
void USBGeckoOutput()
2009-09-25 20:45:51 +02:00
{
LWP_MutexInit(&gecko_mutex, false);
gecko = usb_isgeckoalive(1);
devoptab_list[STD_OUT] = &gecko_out;
devoptab_list[STD_ERR] = &gecko_out;
}
int
main(int argc, char *argv[])
{
2009-11-25 07:35:14 +01:00
USBGeckoOutput(); // uncomment to enable USB gecko output
2009-09-25 20:45:51 +02:00
__exception_setreload(8);
#ifdef HW_DOL
2008-10-21 10:08:30 +02:00
ipl_set_config(6); // disable Qoob modchip
#endif
2008-10-21 10:08:30 +02:00
2009-03-11 18:28:37 +01:00
#ifdef HW_RVL
DI_Init(); // first
#endif
2008-08-20 09:58:55 +02:00
#ifdef DEBUG_WII
2009-03-31 09:16:28 +02:00
//DEBUG_Init(GDBSTUB_DEVICE_USB, 1); // init debugging
//_break();
#endif
2009-02-08 23:30:42 +01:00
InitDeviceThread();
2009-03-11 18:28:37 +01:00
InitGCVideo(); // Initialise video
ResetVideo_Menu (); // change to menu video mode
2009-10-13 06:56:38 +02:00
SetupPads();
2009-03-11 18:28:37 +01:00
#ifdef HW_RVL
// Wii Power/Reset buttons
WPAD_SetPowerButtonCallback((WPADShutdownCallback)ShutdownCB);
SYS_SetPowerCallback(ShutdownCB);
SYS_SetResetCallback(ResetCB);
#endif
2008-10-09 23:44:53 +02:00
// GameCube only - Injected ROM
2008-09-27 09:13:52 +02:00
// Before going any further, let's copy any injected ROM image
2008-10-09 23:44:53 +02:00
// We'll put it in ARAM for safe storage
#ifdef HW_DOL
AR_Init (NULL, 0);
2008-09-27 09:13:52 +02:00
int *romptr = (int *) 0x81000000; // location of injected rom
if (memcmp ((char *) romptr, "SNESROM0", 8) == 0)
{
SNESROMSize = romptr[2];
2008-10-09 23:44:53 +02:00
if(SNESROMSize > (1024*128) && SNESROMSize < (1024*1024*8))
2008-10-09 23:44:53 +02:00
{
romptr = (int *) 0x81000020;
ARAMPut ((char *) romptr, (char *) AR_SNESROM, SNESROMSize);
2008-10-09 23:44:53 +02:00
}
else // not a valid ROM size
{
SNESROMSize = 0;
2008-10-09 23:44:53 +02:00
}
2008-09-27 09:13:52 +02:00
}
2008-10-09 23:44:53 +02:00
#endif
2009-03-11 18:28:37 +01:00
// Initialize DVD subsystem (GameCube only)
#ifdef HW_DOL
DVD_Init ();
#endif
// store path app was loaded from
2009-10-02 00:35:12 +02:00
#ifdef HW_RVL
2008-11-26 08:40:12 +01:00
if(argc > 0 && argv[0] != NULL)
CreateAppPath(argv[0]);
2009-10-02 00:35:12 +02:00
#endif
// Initialize libFAT for SD and USB
2008-12-18 19:36:30 +01:00
MountAllFAT();
// Set defaults
DefaultSettings ();
S9xUnmapAllControls ();
SetDefaultButtonMap ();
2009-11-25 07:35:14 +01:00
S9xReportControllers();
// Allocate SNES Memory
if (!Memory.Init ())
ExitApp();
// Allocate APU
if (!S9xInitAPU ())
ExitApp();
// Set Pixel Renderer to match 565
S9xSetRenderPixelFormat (RGB565);
2009-11-25 07:35:14 +01:00
// Initialise Sound System
S9xInitSound (512, 0);
// Initialise Graphics
setGFX ();
if (!S9xGraphicsInit ())
ExitApp();
S9xInitSync(); // initialize frame sync
2009-03-11 18:28:37 +01:00
// Initialize font system
InitFreeType((u8*)font_ttf, font_ttf_size);
2009-03-11 18:28:37 +01:00
InitGUIThreads();
2008-10-09 23:44:53 +02:00
// GameCube only - Injected ROM
// Everything's been initialized, we can copy our ROM back
// from ARAM into main memory
#ifdef HW_DOL
if(SNESROMSize > 0)
{
ARAMFetchSlow( (char *)Memory.ROM, (char *)AR_SNESROM, SNESROMSize);
2008-10-09 23:44:53 +02:00
Memory.LoadROM ("BLANK.SMC");
Memory.LoadSRAM ("BLANK");
}
2008-10-09 23:44:53 +02:00
#endif
2009-03-11 18:28:37 +01:00
emulate(); // main loop
}