2011-05-28 19:52:00 +00:00
|
|
|
#include <ogc/machine/processor.h>
|
2010-10-27 19:50:48 +00:00
|
|
|
#include "menu/menus.h"
|
2011-01-16 13:12:07 +00:00
|
|
|
#include "menu/WDMMenu.hpp"
|
2010-10-27 19:50:48 +00:00
|
|
|
#include "mload/mload.h"
|
|
|
|
#include "mload/mload_modules.h"
|
2010-11-07 12:16:34 +00:00
|
|
|
#include "system/IosLoader.h"
|
2010-12-30 23:49:22 +00:00
|
|
|
#include "Controls/DeviceHandler.hpp"
|
2010-10-27 19:50:48 +00:00
|
|
|
#include "usbloader/disc.h"
|
2010-11-06 15:30:14 +00:00
|
|
|
#include "usbloader/apploader.h"
|
2011-02-05 21:06:52 +00:00
|
|
|
#include "usbloader/usbstorage2.h"
|
2010-11-06 15:30:14 +00:00
|
|
|
#include "usbloader/wdvd.h"
|
2010-10-27 19:50:48 +00:00
|
|
|
#include "usbloader/GameList.h"
|
|
|
|
#include "settings/Settings.h"
|
|
|
|
#include "settings/CGameSettings.h"
|
|
|
|
#include "usbloader/frag.h"
|
|
|
|
#include "usbloader/wbfs.h"
|
2010-12-27 09:44:27 +00:00
|
|
|
#include "usbloader/playlog.h"
|
2010-12-30 23:49:22 +00:00
|
|
|
#include "usbloader/MountGamePartition.h"
|
2011-01-21 19:43:59 +00:00
|
|
|
#include "usbloader/AlternateDOLOffsets.h"
|
2010-10-27 19:50:48 +00:00
|
|
|
#include "settings/newtitles.h"
|
|
|
|
#include "patches/fst.h"
|
2010-11-06 15:30:14 +00:00
|
|
|
#include "patches/gamepatches.h"
|
|
|
|
#include "patches/wip.h"
|
|
|
|
#include "system/IosLoader.h"
|
2010-12-27 09:44:27 +00:00
|
|
|
#include "banner/OpeningBNR.hpp"
|
2010-10-27 19:50:48 +00:00
|
|
|
#include "wad/nandtitle.h"
|
2010-11-06 15:30:14 +00:00
|
|
|
#include "menu/menus.h"
|
|
|
|
#include "memory/memory.h"
|
2011-01-22 00:06:28 +00:00
|
|
|
#include "GameBooter.hpp"
|
2010-11-06 15:30:14 +00:00
|
|
|
#include "sys.h"
|
|
|
|
|
|
|
|
//appentrypoint has to be global because of asm
|
|
|
|
u32 AppEntrypoint = 0;
|
2010-10-27 19:50:48 +00:00
|
|
|
|
|
|
|
struct discHdr *dvdheader = NULL;
|
2011-01-28 21:19:43 +00:00
|
|
|
extern u32 hdd_sector_size;
|
2010-10-27 19:50:48 +00:00
|
|
|
extern int mountMethod;
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
int GameBooter::BootGCMode()
|
|
|
|
{
|
|
|
|
ExitApp();
|
|
|
|
gprintf("\nLoading BC for GameCube");
|
|
|
|
WII_Initialize();
|
|
|
|
return WII_LaunchTitle(0x0000000100000100ULL);
|
|
|
|
}
|
2010-11-06 15:30:14 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
|
|
|
|
u32 GameBooter::BootPartition(char * dolpath, u8 videoselected, u8 languageChoice, u8 cheat, u8 vipatch, u8 patchcountrystring,
|
|
|
|
u8 alternatedol, u32 alternatedoloffset, u32 returnTo, u8 fix002)
|
2010-11-06 15:30:14 +00:00
|
|
|
{
|
2011-01-16 13:12:07 +00:00
|
|
|
gprintf("booting partition IOS %u r%u\n", IOS_GetVersion(), IOS_GetRevision());
|
2010-11-06 15:30:14 +00:00
|
|
|
entry_point p_entry;
|
|
|
|
s32 ret;
|
|
|
|
u64 offset;
|
|
|
|
|
|
|
|
/* Find game partition offset */
|
2011-03-14 21:19:21 +00:00
|
|
|
ret = Disc_FindPartition(&offset);
|
2010-11-06 15:30:14 +00:00
|
|
|
if (ret < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Open specified partition */
|
|
|
|
ret = WDVD_OpenPartition(offset);
|
|
|
|
if (ret < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
load_wip_code((u8*) Disc_ID);
|
|
|
|
|
|
|
|
/* If a wip file is loaded for this game this does nothing - Dimok */
|
|
|
|
PoPPatch();
|
2011-01-09 17:51:15 +00:00
|
|
|
NSMBPatch();
|
2010-11-06 15:30:14 +00:00
|
|
|
|
|
|
|
/* Setup low memory */
|
2011-03-14 21:19:21 +00:00
|
|
|
Disc_SetLowMem();
|
|
|
|
|
|
|
|
/* Setup video mode */
|
|
|
|
Disc_SelectVMode(videoselected);
|
2010-11-06 15:30:14 +00:00
|
|
|
|
|
|
|
/* Run apploader */
|
|
|
|
ret = Apploader_Run(&p_entry, dolpath, cheat, videoselected, languageChoice, vipatch, patchcountrystring,
|
|
|
|
alternatedol, alternatedoloffset, returnTo, fix002);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
free_wip();
|
|
|
|
|
|
|
|
return (u32) p_entry;
|
|
|
|
}
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
int GameBooter::FindDiscHeader(const char * gameID, struct discHdr &gameHeader)
|
2010-10-27 19:50:48 +00:00
|
|
|
{
|
|
|
|
gameList.LoadUnfiltered();
|
|
|
|
|
2010-12-26 17:02:14 +00:00
|
|
|
if(mountMethod == 0 && !gameList.GetDiscHeader(gameID))
|
2010-10-27 19:50:48 +00:00
|
|
|
{
|
|
|
|
gprintf("Game was not found: %s\n", gameID);
|
|
|
|
return -1;
|
|
|
|
}
|
2010-12-26 17:02:14 +00:00
|
|
|
else if(mountMethod && !dvdheader)
|
|
|
|
{
|
|
|
|
gprintf("Error: Loading empty disc header from DVD\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2010-10-27 19:50:48 +00:00
|
|
|
|
2010-12-26 17:02:14 +00:00
|
|
|
memcpy(&gameHeader, (mountMethod ? dvdheader : gameList.GetDiscHeader(gameID)), sizeof(struct discHdr));
|
2010-11-28 20:04:29 +00:00
|
|
|
|
2010-12-26 17:02:14 +00:00
|
|
|
delete dvdheader;
|
|
|
|
dvdheader = NULL;
|
2011-01-12 19:30:04 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameBooter::SetupAltDOL(u8 * gameID, u8 &alternatedol, u32 &alternatedoloffset)
|
|
|
|
{
|
|
|
|
if(alternatedol == ALT_DOL_ON_LAUNCH)
|
|
|
|
{
|
|
|
|
alternatedol = ALT_DOL_FROM_GAME;
|
|
|
|
alternatedoloffset = WDMMenu::GetAlternateDolOffset();
|
|
|
|
}
|
|
|
|
else if(alternatedol == ALT_DOL_DEFAULT)
|
|
|
|
{
|
|
|
|
alternatedol = ALT_DOL_FROM_GAME;
|
|
|
|
alternatedoloffset = defaultAltDol((char *) gameID);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(alternatedol == ALT_DOL_FROM_GAME && alternatedoloffset == 0)
|
|
|
|
alternatedol = OFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
int GameBooter::SetupDisc(u8 * gameID)
|
|
|
|
{
|
|
|
|
if (mountMethod)
|
|
|
|
{
|
|
|
|
gprintf("\tloading DVD\n");
|
|
|
|
return Disc_Open();
|
|
|
|
}
|
|
|
|
|
|
|
|
int ret = -1;
|
|
|
|
|
2011-01-28 21:19:43 +00:00
|
|
|
if(((IosLoader::IsWaninkokoIOS() && IOS_GetRevision() < 18) ||
|
2011-02-02 18:30:15 +00:00
|
|
|
hdd_sector_size != 512) && gameList.GetGameFS(gameID) == PART_FS_WBFS)
|
2011-01-22 00:06:28 +00:00
|
|
|
{
|
|
|
|
gprintf("Disc_SetUSB...");
|
|
|
|
ret = Disc_SetUSB(gameID);
|
|
|
|
gprintf("%d\n", ret);
|
|
|
|
if(ret < 0) return ret;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gprintf("Loading fragment list...");
|
|
|
|
ret = get_frag_list(gameID);
|
|
|
|
gprintf("%d\n", ret);
|
|
|
|
if(ret < 0) return ret;
|
|
|
|
ret = set_frag_list(gameID);
|
|
|
|
if(ret < 0) return ret;
|
|
|
|
gprintf("\tUSB set to game\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
gprintf("Disc_Open()...");
|
|
|
|
ret = Disc_Open();
|
|
|
|
gprintf("%d\n", ret);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GameBooter::LoadOcarina(u8 *gameID)
|
|
|
|
{
|
|
|
|
if (ocarina_load_code(gameID) > 0)
|
|
|
|
{
|
|
|
|
ocarina_do_code();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int GameBooter::BootGame(const char * gameID)
|
|
|
|
{
|
|
|
|
if(!gameID || strlen(gameID) < 3)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (mountMethod == 2)
|
|
|
|
return BootGCMode();
|
|
|
|
|
|
|
|
AppCleanUp();
|
2010-10-27 19:50:48 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
gprintf("\tSettings.partition: %d\n", Settings.partition);
|
|
|
|
|
|
|
|
struct discHdr gameHeader;
|
|
|
|
|
|
|
|
//! Find disc header in the game list first
|
|
|
|
int ret = FindDiscHeader(gameID, gameHeader);
|
|
|
|
if(ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
//! Setup game configuration from game settings. If no game settings exist use global/default.
|
2010-12-26 17:02:14 +00:00
|
|
|
GameCFG * game_cfg = GameSettings.GetGameCFG(gameHeader.id);
|
2011-05-28 19:52:00 +00:00
|
|
|
u8 videoChoice = game_cfg->video == INHERIT ? Settings.videomode : game_cfg->video;
|
|
|
|
u8 languageChoice = game_cfg->language == INHERIT ? Settings.language : game_cfg->language;
|
|
|
|
u8 ocarinaChoice = game_cfg->ocarina == INHERIT ? Settings.ocarina : game_cfg->ocarina;
|
|
|
|
u8 viChoice = game_cfg->vipatch == INHERIT ? Settings.videopatch : game_cfg->vipatch;
|
|
|
|
u8 iosChoice = game_cfg->ios == INHERIT ? Settings.cios : game_cfg->ios;
|
|
|
|
u8 fix002 = game_cfg->errorfix002 == INHERIT ? Settings.error002 : game_cfg->errorfix002;
|
|
|
|
u8 countrystrings = game_cfg->patchcountrystrings == INHERIT ? Settings.patchcountrystrings : game_cfg->patchcountrystrings;
|
2011-01-21 19:43:59 +00:00
|
|
|
u8 alternatedol = game_cfg->loadalternatedol;
|
|
|
|
u32 alternatedoloffset = game_cfg->alternatedolstart;
|
|
|
|
u8 reloadblock = game_cfg->iosreloadblock;
|
|
|
|
u8 returnToLoaderGV = game_cfg->returnTo;
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Prepare alternate dol settings
|
|
|
|
SetupAltDOL(gameHeader.id, alternatedol, alternatedoloffset);
|
|
|
|
|
|
|
|
//! Setup the return to Loader option
|
|
|
|
u32 channel = 0;
|
|
|
|
if (returnToLoaderGV)
|
2011-01-21 19:43:59 +00:00
|
|
|
{
|
2011-01-22 00:06:28 +00:00
|
|
|
int idx = NandTitles.FindU32(Settings.returnTo);
|
2011-05-28 19:52:00 +00:00
|
|
|
|
|
|
|
//! this is here for test purpose only and needs be moved later
|
|
|
|
static char es_fs[] ATTRIBUTE_ALIGN(32) = "/dev/es";
|
|
|
|
static u64 sm_title_id ATTRIBUTE_ALIGN(32);
|
|
|
|
STACK_ALIGN(ioctlv, vector, 1, 32);
|
|
|
|
|
|
|
|
sm_title_id = NandTitles.At(idx);
|
|
|
|
vector[0].data = &sm_title_id;
|
|
|
|
vector[0].len = sizeof(sm_title_id);
|
|
|
|
|
|
|
|
int es_fd = IOS_Open(es_fs, 0);
|
|
|
|
int result = -1;
|
|
|
|
|
|
|
|
if(es_fd >= 0)
|
|
|
|
result = IOS_Ioctlv(es_fd, 0xA1, 1, 0, vector);
|
|
|
|
|
|
|
|
if(es_fd >= 0)
|
|
|
|
IOS_Close(es_fd);
|
|
|
|
|
|
|
|
//! use old method in case of failure for other IOS versions than d2x
|
|
|
|
if (result < 0 && idx >= 0)
|
|
|
|
channel = TITLE_LOWER(NandTitles.At(idx));
|
2011-01-21 19:43:59 +00:00
|
|
|
}
|
2011-01-16 13:12:07 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! This is temporary - C <-> C++ transfer
|
|
|
|
SetCheatFilepath(Settings.Cheatcodespath);
|
|
|
|
SetBCAFilepath(Settings.BcaCodepath);
|
|
|
|
|
|
|
|
//! Reload game settings cIOS for this game
|
2010-11-06 15:30:14 +00:00
|
|
|
if(iosChoice != IOS_GetVersion())
|
|
|
|
{
|
2010-11-07 12:16:34 +00:00
|
|
|
gprintf("Reloading into game cIOS: %i...\n", iosChoice);
|
2010-11-06 15:30:14 +00:00
|
|
|
IosLoader::LoadGameCios(iosChoice);
|
|
|
|
if(MountGamePartition(false) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Setup disc in cIOS and open it
|
|
|
|
ret = SetupDisc(gameHeader.id);
|
2010-11-07 12:16:34 +00:00
|
|
|
if (ret < 0)
|
|
|
|
Sys_BackToLoader();
|
2010-10-27 19:50:48 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Load BCA data for the game
|
2010-10-27 19:50:48 +00:00
|
|
|
gprintf("Loading BCA data...");
|
2010-12-26 17:02:14 +00:00
|
|
|
ret = do_bca_code(gameHeader.id);
|
2010-10-27 19:50:48 +00:00
|
|
|
gprintf("%d\n", ret);
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Setup IOS reload block - only possible on Hermes cIOS
|
2010-11-07 12:16:34 +00:00
|
|
|
if (reloadblock == ON && IosLoader::IsHermesIOS())
|
2010-10-27 19:50:48 +00:00
|
|
|
{
|
|
|
|
enable_ES_ioctlv_vector();
|
2011-02-02 18:30:15 +00:00
|
|
|
if (gameList.GetGameFS(gameHeader.id) == PART_FS_WBFS)
|
2010-10-27 19:50:48 +00:00
|
|
|
mload_close();
|
|
|
|
}
|
2011-02-05 21:06:52 +00:00
|
|
|
|
2011-02-02 18:30:15 +00:00
|
|
|
//! Now we can free up the memory used by the game list
|
|
|
|
gameList.clear();
|
2010-10-27 19:50:48 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Load main.dol or alternative dol into memory, start the game apploader and get game entrypoint
|
2010-10-27 19:50:48 +00:00
|
|
|
gprintf("\tDisc_wiiBoot\n");
|
2010-11-06 15:30:14 +00:00
|
|
|
AppEntrypoint = BootPartition(Settings.dolpath, videoChoice, languageChoice, ocarinaChoice, viChoice, countrystrings,
|
2010-10-27 19:50:48 +00:00
|
|
|
alternatedol, alternatedoloffset, channel, fix002);
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! No entrypoint found...back to HBC/SystemMenu
|
2011-01-21 19:43:59 +00:00
|
|
|
if(AppEntrypoint == 0)
|
2010-11-06 15:30:14 +00:00
|
|
|
{
|
2011-01-21 19:43:59 +00:00
|
|
|
WDVD_ClosePartition();
|
|
|
|
Sys_BackToLoader();
|
|
|
|
}
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Load Ocarina codes
|
2011-01-21 19:43:59 +00:00
|
|
|
bool enablecheat = false;
|
|
|
|
if (ocarinaChoice)
|
2011-01-22 00:06:28 +00:00
|
|
|
enablecheat = LoadOcarina(gameHeader.id);
|
2010-11-06 15:30:14 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Shadow mload - Only needed on some games with Hermes v5.1 (Check is inside the function)
|
2011-01-21 19:43:59 +00:00
|
|
|
shadow_mload();
|
2011-01-22 00:06:28 +00:00
|
|
|
|
2011-02-05 21:06:52 +00:00
|
|
|
//! Remember game's USB port
|
|
|
|
int usbport = USBStorage2_GetPort();
|
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Flush all caches and close up all devices
|
2011-02-02 18:30:15 +00:00
|
|
|
WBFS_CloseAll();
|
2011-01-21 19:43:59 +00:00
|
|
|
DeviceHandler::DestroyInstance();
|
2011-02-05 21:06:52 +00:00
|
|
|
if(Settings.USBPort == 2)
|
|
|
|
{
|
|
|
|
//! Reset USB port because device handler changes it for cache flushing
|
|
|
|
USBStorage2_Init();
|
|
|
|
USBStorage2_SetPort(usbport);
|
|
|
|
USBStorage2_Deinit();
|
|
|
|
}
|
2011-05-28 19:52:00 +00:00
|
|
|
USB_Deinitialize();
|
2011-02-05 21:06:52 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Modify Wii Message Board to display the game starting here
|
2011-01-21 19:43:59 +00:00
|
|
|
if(Settings.PlaylogUpdate)
|
2011-01-22 00:06:28 +00:00
|
|
|
Playlog_Update((char *) gameHeader.id, BNRInstance::Instance()->GetIMETTitle(CONF_GetLanguage()));
|
2010-10-27 19:50:48 +00:00
|
|
|
|
2011-01-22 00:06:28 +00:00
|
|
|
//! Jump to the entrypoint of the game - the last function of the USB Loader
|
2011-01-21 19:43:59 +00:00
|
|
|
gprintf("Jumping to game entrypoint: 0x%08X.\n", AppEntrypoint);
|
2011-03-14 21:19:21 +00:00
|
|
|
return Disc_JumpToEntrypoint(enablecheat, WDMMenu::GetDolParameter());
|
2010-10-27 19:50:48 +00:00
|
|
|
}
|