vbagx/source/vbagx.cpp

451 lines
9.7 KiB
C++
Raw Normal View History

/****************************************************************************
2008-09-17 04:27:55 +02:00
* Visual Boy Advance GX
*
* Tantric September 2008
*
2010-03-31 04:45:09 +02:00
* vbagx.cpp
2008-09-17 04:27:55 +02:00
*
* 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>
2009-09-25 20:50:57 +02:00
#include <sys/iosupport.h>
#ifdef HW_RVL
#include <di/di.h>
#endif
2010-03-22 00:49:59 +01:00
#include "vbagx.h"
#include "vbasupport.h"
#include "preferences.h"
#include "audio.h"
2008-12-24 08:52:35 +01:00
#include "networkop.h"
2009-04-08 09:08:12 +02:00
#include "filebrowser.h"
#include "fileop.h"
#include "menu.h"
#include "input.h"
#include "video.h"
#include "gamesettings.h"
#include "mem2.h"
2010-03-22 00:49:59 +01:00
#include "utils/usb2storage.h"
#include "utils/mload.h"
#include "utils/FreeTypeGX.h"
#include "vba/gba/Globals.h"
#include "vba/gba/Sound.h"
2009-09-25 20:46:25 +02:00
extern "C" {
extern void __exception_setreload(int t);
}
extern int emulating;
void StopColorizing();
void gbSetPalette(u32 RRGGBB[]);
int ScreenshotRequested = 0;
2008-10-18 22:40:48 +02:00
int ConfigRequested = 0;
int ShutdownRequested = 0;
int ResetRequested = 0;
2009-04-08 09:08:12 +02:00
int ExitRequested = 0;
2009-10-02 00:42:02 +02:00
char appPath[1024] = { 0 };
char loadedFile[1024] = { 0 };
extern FILE *out;
/****************************************************************************
* Shutdown / Reboot / Exit
***************************************************************************/
static void ExitCleanup()
2008-12-18 19:58:30 +01:00
{
2009-04-08 09:08:12 +02:00
ShutdownAudio();
StopGX();
if (out) fclose(out);
2009-04-08 09:08:12 +02:00
2009-05-06 19:28:50 +02:00
HaltDeviceThread();
2008-12-18 19:58: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-04-08 09:08:12 +02:00
void ExitApp()
{
2009-10-06 09:17:52 +02:00
#ifdef HW_RVL
ShutoffRumble();
#endif
SavePrefs(SILENT);
if (ROMLoaded && !ConfigRequested && GCSettings.AutoSave == 1)
2009-10-02 00:42:02 +02:00
SaveBatteryOrStateAuto(FILE_SRAM, SILENT);
2008-12-18 19:58:30 +01:00
ExitCleanup();
if(ShutdownRequested)
SYS_ResetSystem(SYS_POWEROFF, 0, 0);
#ifdef HW_RVL
if(GCSettings.ExitAction == 0) // Auto
2009-04-08 09:08:12 +02: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-04-08 09:08:12 +02:00
}
#endif
if(GCSettings.ExitAction == 1) // Exit to Menu
2009-04-08 09:08:12 +02: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-04-08 09:08:12 +02: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()
{
2010-04-17 05:06:12 +02:00
if(!inNetworkInit)
ShutdownRequested = 1;
}
void ResetCB()
{
2010-04-17 05:06:12 +02:00
if(!inNetworkInit)
ResetRequested = 1;
}
#endif
#ifdef HW_DOL
2008-10-21 10:09:18 +02:00
/****************************************************************************
* ipl_set_config
* lowlevel Qoob Modchip disable
***************************************************************************/
static void ipl_set_config(unsigned char c)
2008-10-21 10:09:18 +02:00
{
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
2008-10-21 10:09:18 +02:00
/****************************************************************************
* IOS 202
***************************************************************************/
#ifdef HW_RVL
static bool FindIOS(u32 ios)
{
s32 ret;
u32 n;
u64 *titles = NULL;
u32 num_titles=0;
ret = ES_GetNumTitles(&num_titles);
if (ret < 0)
return false;
if(num_titles < 1)
return false;
titles = (u64 *)memalign(32, num_titles * sizeof(u64) + 32);
if (!titles)
return false;
ret = ES_GetTitles(titles, num_titles);
if (ret < 0)
{
free(titles);
return false;
}
for(n=0; n < num_titles; n++)
{
if((titles[n] & 0xFFFFFFFF)==ios)
{
free(titles);
return true;
}
}
free(titles);
return false;
}
#define ROUNDDOWN32(v) (((u32)(v)-0x1f)&~0x1f)
static bool USBLANDetected()
{
u8 dummy;
u8 i;
u16 vid, pid;
USB_Initialize();
u8 *buffer = (u8*)ROUNDDOWN32(((u32)SYS_GetArena2Hi() - (32*1024)));
memset(buffer, 0, 8 << 3);
if(USB_GetDeviceList("/dev/usb/oh0", buffer, 8, 0, &dummy) < 0)
return false;
for(i = 0; i < 8; i++)
{
memcpy(&vid, (buffer + (i << 3) + 4), 2);
memcpy(&pid, (buffer + (i << 3) + 6), 2);
if( (vid == 0x0b95) && (pid == 0x7720))
return true;
}
return false;
}
#endif
2009-09-25 20:46:25 +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:55 +02:00
void USBGeckoOutput()
2009-09-25 20:46:25 +02:00
{
LWP_MutexInit(&gecko_mutex, false);
gecko = usb_isgeckoalive(1);
devoptab_list[STD_OUT] = &gecko_out;
devoptab_list[STD_ERR] = &gecko_out;
}
/****************************************************************************
* main
*
* Program entry
****************************************************************************/
int main(int argc, char *argv[])
{
2009-09-25 20:46:25 +02:00
//USBGeckoOutput(); // uncomment to enable USB gecko output
__exception_setreload(8);
2008-10-21 10:09:18 +02:00
#ifdef HW_DOL
ipl_set_config(6); // disable Qoob modchip
#endif
#ifdef HW_RVL
bool usblan = USBLANDetected();
// try to load IOS 202
2010-06-05 03:45:27 +02:00
if(FindIOS(202))
IOS_ReloadIOS(202);
2010-06-05 03:45:27 +02:00
else if(IOS_GetVersion() < 61 && FindIOS(61))
IOS_ReloadIOS(61);
2010-03-11 19:18:14 +01:00
if(IOS_GetVersion() == 202)
{
// enable DVD and USB2
2010-03-11 19:18:14 +01:00
if(mload_init() >= 0 && load_ehci_module())
{
int mode = 3;
if(usblan)
{
int usblanport = GetUSB2LanPort();
if(usblanport >= 0)
{
if(usblanport == 1)
mode = 1;
else
mode = 2;
USB2Storage_Close();
mload_close();
IOS_ReloadIOS(202);
mload_init();
load_ehci_module();
}
}
SetUSB2Mode(mode);
2010-03-11 19:18:14 +01:00
USB2Enable(true);
DI_LoadDVDX(false);
DI_Init();
}
2010-03-11 19:18:14 +01:00
}
#endif
InitDeviceThread();
2009-04-08 09:08:12 +02:00
InitializeVideo();
2009-02-20 10:36:05 +01:00
ResetVideo_Menu (); // change to menu video mode
2009-10-13 07:05:16 +02:00
SetupPads();
2010-06-08 20:00:58 +02:00
// Initialize DVD subsystem (GameCube only)
#ifdef HW_DOL
DVD_Init ();
#endif
2009-02-20 10:36:05 +01:00
#ifdef HW_RVL
// Wii Power/Reset buttons
WPAD_SetPowerButtonCallback((WPADShutdownCallback)ShutdownCB);
SYS_SetPowerCallback(ShutdownCB);
SYS_SetResetCallback(ResetCB);
#endif
2009-04-08 09:08:12 +02:00
MountAllFAT(); // Initialize libFAT for SD and USB
InitialiseSound();
InitialisePalette();
2009-04-08 09:08:12 +02:00
DefaultSettings (); // Set defaults
InitFreeType((u8*)font_ttf, font_ttf_size); // Initialize font system
2009-10-02 00:42:02 +02:00
#ifdef HW_RVL
2008-11-26 08:39:09 +01:00
if(argc > 0 && argv[0] != NULL)
CreateAppPath(argv[0]); // store path app was loaded from
InitMem2Manager();
savebuffer = (unsigned char *)mem2_malloc(SAVEBUFFERSIZE);
browserList = (BROWSERENTRY *)mem2_malloc(sizeof(BROWSERENTRY)*MAX_BROWSER_SIZE);
rom = (u8 *)mem2_malloc(1024*1024*32); // allocate 32 MB to GBA ROM
#else
savebuffer = (unsigned char *)malloc(SAVEBUFFERSIZE);
browserList = (BROWSERENTRY *)malloc(sizeof(BROWSERENTRY)*MAX_BROWSER_SIZE);
2009-10-02 00:42:02 +02:00
#endif
gameScreenPng = (u8 *)malloc(512*1024);
InitGUIThreads();
2008-10-21 09:17:07 +02:00
while(1) // main loop
{
// go back to checking if devices were inserted/removed
// since we're entering the menu
2009-05-06 19:28:50 +02:00
ResumeDeviceThread();
2009-04-08 09:08:12 +02:00
SwitchAudioMode(1);
if(!ROMLoaded)
MainMenu(MENU_GAMESELECTION);
else
MainMenu(MENU_GAME);
ConfigRequested = 0;
ScreenshotRequested = 0;
2009-04-08 09:08:12 +02:00
SwitchAudioMode(0);
2008-10-18 22:40:48 +02:00
// stop checking if devices were removed/inserted
// since we're starting emulation again
2009-05-06 19:28:50 +02:00
HaltDeviceThread();
2008-10-24 08:16:15 +02:00
ResetVideo_Emu();
// GB colorizing - set palette
if(IsGameboyGame())
{
if(GCSettings.colorize && strcmp(RomTitle, "MEGAMAN") != 0)
gbSetPalette(CurrentPalette.palette);
else
StopColorizing();
}
2008-10-21 09:17:07 +02:00
while (emulating) // emulation loop
{
emulator.emuMain(emulator.emuCount);
2008-10-18 22:40:48 +02:00
if(ResetRequested)
{
emulator.emuReset(); // reset game
ResetRequested = 0;
}
2008-10-18 22:40:48 +02:00
if(ConfigRequested)
{
2009-04-08 09:08:12 +02:00
ResetVideo_Menu();
break; // leave emulation loop
2008-10-18 22:40:48 +02:00
}
#ifdef HW_RVL
if(ShutdownRequested)
ExitApp();
#endif
2009-04-08 09:08:12 +02:00
} // emulation loop
} // main loop
return 0;
}