2012-01-21 22:15:45 +01:00
|
|
|
#include <gccore.h>
|
2012-02-25 23:34:52 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
2012-03-11 20:47:24 +01:00
|
|
|
#include <string.h>
|
2012-01-23 16:57:30 +01:00
|
|
|
#include "gc.h"
|
2012-02-25 23:34:52 +01:00
|
|
|
#include "gecko.h"
|
2012-02-27 17:05:10 +01:00
|
|
|
#include "fileOps.h"
|
2012-02-28 19:44:41 +01:00
|
|
|
#include "utils.h"
|
2012-03-11 20:47:24 +01:00
|
|
|
#include "memory/mem2.hpp"
|
2012-01-21 22:15:45 +01:00
|
|
|
|
2012-07-12 15:21:14 +02:00
|
|
|
// for directory parsing and low-level file I/O
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <dirent.h>
|
|
|
|
|
|
|
|
// DIOS-MIOS
|
|
|
|
DML_CFG *DMLCfg = NULL;
|
|
|
|
|
2012-07-12 17:53:04 +02:00
|
|
|
void DML_New_SetOptions(const char *GamePath, char *CheatPath, char *NewCheatPath, bool cheats, bool debugger, u8 NMM, u8 nodisc, u8 DMLvideoMode, u8 videoSetting)
|
2012-07-12 15:21:14 +02:00
|
|
|
{
|
|
|
|
gprintf("Wiiflow DML: Launch game '%s' through memory (new method)\n", GamePath);
|
|
|
|
|
|
|
|
DMLCfg = (DML_CFG*)MEM1_alloc(sizeof(DML_CFG));
|
|
|
|
if(DMLCfg == NULL)
|
|
|
|
return;
|
|
|
|
memset(DMLCfg, 0, sizeof(DML_CFG));
|
|
|
|
|
|
|
|
DMLCfg->Magicbytes = 0xD1050CF6;
|
|
|
|
DMLCfg->CfgVersion = 0x00000001;
|
2012-07-12 17:53:04 +02:00
|
|
|
if(videoSetting == 0)
|
|
|
|
DMLCfg->VideoMode |= DML_VID_NONE;
|
|
|
|
else if(videoSetting == 1)
|
2012-07-12 15:21:14 +02:00
|
|
|
DMLCfg->VideoMode |= DML_VID_DML_AUTO;
|
2012-07-12 17:53:04 +02:00
|
|
|
else
|
|
|
|
DMLCfg->VideoMode |= DML_VID_FORCE;
|
2012-07-12 15:21:14 +02:00
|
|
|
|
|
|
|
DMLCfg->Config |= DML_CFG_ACTIVITY_LED; //Sorry but I like it lol, option will may follow
|
|
|
|
DMLCfg->Config |= DML_CFG_PADHOOK; //Makes life easier, l+z+b+digital down...
|
|
|
|
|
|
|
|
if(GamePath != NULL)
|
|
|
|
{
|
|
|
|
strncpy(DMLCfg->GamePath, GamePath, sizeof(DMLCfg->GamePath));
|
|
|
|
DMLCfg->Config |= DML_CFG_GAME_PATH;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(CheatPath != NULL && NewCheatPath != NULL && cheats)
|
|
|
|
{
|
|
|
|
char *ptr;
|
|
|
|
if(strstr(CheatPath, "sd:/") == NULL)
|
|
|
|
{
|
|
|
|
fsop_CopyFile(CheatPath, NewCheatPath, NULL, NULL);
|
|
|
|
ptr = &NewCheatPath[3];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ptr = &CheatPath[3];
|
|
|
|
strncpy(DMLCfg->CheatPath, ptr, sizeof(DMLCfg->CheatPath));
|
|
|
|
DMLCfg->Config |= DML_CFG_CHEAT_PATH;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(cheats)
|
|
|
|
DMLCfg->Config |= DML_CFG_CHEATS;
|
|
|
|
if(debugger)
|
|
|
|
DMLCfg->Config |= DML_CFG_DEBUGGER;
|
|
|
|
if(NMM > 0)
|
|
|
|
DMLCfg->Config |= DML_CFG_NMM;
|
|
|
|
if(NMM > 1)
|
|
|
|
DMLCfg->Config |= DML_CFG_NMM_DEBUG;
|
|
|
|
if(nodisc > 0)
|
|
|
|
DMLCfg->Config |= DML_CFG_NODISC;
|
|
|
|
|
|
|
|
if(DMLvideoMode > 3)
|
|
|
|
DMLCfg->VideoMode |= DML_VID_PROG_PATCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DML_Old_SetOptions(char *GamePath, char *CheatPath, char *NewCheatPath, bool cheats)
|
|
|
|
{
|
|
|
|
gprintf("Wiiflow DML: Launch game '%s' through boot.bin (old method)\n", GamePath);
|
|
|
|
FILE *f;
|
|
|
|
f = fopen("sd:/games/boot.bin", "wb");
|
|
|
|
fwrite(GamePath, 1, strlen(GamePath) + 1, f);
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
if(cheats && strstr(CheatPath, NewCheatPath) == NULL)
|
|
|
|
fsop_CopyFile(CheatPath, NewCheatPath, NULL, NULL);
|
|
|
|
|
|
|
|
//Tell DML to boot the game from sd card
|
|
|
|
*(vu32*)0x80001800 = 0xB002D105;
|
|
|
|
DCFlushRange((void *)(0x80001800), 4);
|
|
|
|
ICInvalidateRange((void *)(0x80001800), 4);
|
|
|
|
|
|
|
|
*(vu32*)0xCC003024 |= 7;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DML_New_SetBootDiscOption()
|
|
|
|
{
|
|
|
|
gprintf("Booting GC game\n");
|
|
|
|
|
|
|
|
DMLCfg = (DML_CFG*)MEM1_alloc(sizeof(DML_CFG));
|
|
|
|
if(DMLCfg == NULL)
|
|
|
|
return;
|
|
|
|
memset(DMLCfg, 0, sizeof(DML_CFG));
|
|
|
|
|
|
|
|
DMLCfg->Magicbytes = 0xD1050CF6;
|
|
|
|
DMLCfg->CfgVersion = 0x00000001;
|
|
|
|
DMLCfg->VideoMode |= DML_VID_DML_AUTO;
|
|
|
|
|
|
|
|
DMLCfg->Config |= DML_CFG_BOOT_DISC;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DML_New_WriteOptions()
|
|
|
|
{
|
|
|
|
if(DMLCfg == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
//Write options into memory
|
|
|
|
memcpy((void *)0x80001700, DMLCfg, sizeof(DML_CFG));
|
|
|
|
DCFlushRange((void *)(0x80001700), sizeof(DML_CFG));
|
|
|
|
|
|
|
|
//DML v1.2+
|
|
|
|
memcpy((void *)0x81200000, DMLCfg, sizeof(DML_CFG));
|
|
|
|
DCFlushRange((void *)(0x81200000), sizeof(DML_CFG));
|
|
|
|
|
|
|
|
MEM1_free(DMLCfg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Devolution
|
2012-07-12 18:02:51 +02:00
|
|
|
u8 *loader_bin = NULL;
|
2012-07-12 15:21:14 +02:00
|
|
|
static gconfig *DEVO_CONFIG = (gconfig*)0x80000020;
|
2012-07-12 18:02:51 +02:00
|
|
|
#define LAUNCH() ((void(*)(void))loader_bin)()
|
2012-07-12 15:21:14 +02:00
|
|
|
|
2012-07-12 17:53:04 +02:00
|
|
|
bool DEVO_Installed(const char* path)
|
|
|
|
{
|
|
|
|
bool devo = false;
|
|
|
|
char loader_path[256];
|
|
|
|
snprintf(loader_path, sizeof(loader_path), "%s/loader.bin", path);
|
|
|
|
FILE *f = fopen(loader_path, "rb");
|
|
|
|
if(f)
|
|
|
|
{
|
|
|
|
devo = true;
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
return devo;
|
|
|
|
}
|
|
|
|
|
2012-07-12 18:02:51 +02:00
|
|
|
void DEVO_SetOptions(const char *path, const char *partition, const char* loader)
|
|
|
|
{
|
|
|
|
//Read in loader.bin
|
|
|
|
char loader_path[256];
|
|
|
|
snprintf(loader_path, sizeof(loader_path), "%s/loader.bin", loader);
|
|
|
|
FILE *f = fopen(loader_path, "rb");
|
|
|
|
if(f)
|
|
|
|
{
|
|
|
|
fseek(f, 0, SEEK_END);
|
|
|
|
u32 size = ftell(f);
|
|
|
|
rewind(f);
|
|
|
|
loader_bin = (u8*)MEM2_alloc(size);
|
|
|
|
fread(loader_bin, 1, size, f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
//start writing cfg to mem
|
2012-07-12 15:21:14 +02:00
|
|
|
struct stat st;
|
|
|
|
char full_path[256];
|
|
|
|
int data_fd;
|
|
|
|
|
|
|
|
stat(path, &st);
|
|
|
|
u8 *lowmem = (u8*)0x80000000;
|
|
|
|
FILE *iso_file = fopen(path, "rb");
|
|
|
|
fread(lowmem, 1, 32, iso_file);
|
|
|
|
fclose(iso_file);
|
|
|
|
|
|
|
|
// fill out the Devolution config struct
|
|
|
|
memset(DEVO_CONFIG, 0, sizeof(*DEVO_CONFIG));
|
|
|
|
DEVO_CONFIG->signature = 0x3EF9DB23;
|
|
|
|
DEVO_CONFIG->version = 0x00000100;
|
|
|
|
DEVO_CONFIG->device_signature = st.st_dev;
|
|
|
|
DEVO_CONFIG->disc1_cluster = st.st_ino;
|
|
|
|
|
|
|
|
// make sure these directories exist, they are required for Devolution to function correctly
|
|
|
|
snprintf(full_path, sizeof(full_path), "%s:/apps", partition);
|
|
|
|
fsop_MakeFolder(full_path);
|
|
|
|
snprintf(full_path, sizeof(full_path), "%s:/apps/gc_devo", partition);
|
|
|
|
fsop_MakeFolder(full_path);
|
|
|
|
|
|
|
|
// find or create a 16MB memcard file for emulation
|
|
|
|
// this file can be located anywhere since it's passed by cluster, not name
|
|
|
|
// it must be at least 16MB though
|
|
|
|
snprintf(full_path, sizeof(full_path), "%s:/apps/gc_devo/memcard.bin", partition);
|
|
|
|
|
|
|
|
// check if file doesn't exist or is less than 16MB
|
|
|
|
if(stat(full_path, &st) == -1 || st.st_size < 16<<20)
|
|
|
|
{
|
|
|
|
// need to enlarge or create it
|
|
|
|
data_fd = open(full_path, O_WRONLY|O_CREAT);
|
|
|
|
if (data_fd>=0)
|
|
|
|
{
|
|
|
|
// make it 16MB
|
|
|
|
printf("Resizing memcard file...\n");
|
|
|
|
ftruncate(data_fd, 16<<20);
|
|
|
|
if (fstat(data_fd, &st)==-1 || st.st_size < 16<<20)
|
|
|
|
{
|
|
|
|
// it still isn't big enough. Give up.
|
|
|
|
st.st_ino = 0;
|
|
|
|
}
|
|
|
|
close(data_fd);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// couldn't open or create the memory card file
|
|
|
|
st.st_ino = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// set FAT cluster for start of memory card file
|
|
|
|
// if this is zero memory card emulation will not be used
|
|
|
|
DEVO_CONFIG->memcard_cluster = st.st_ino;
|
|
|
|
|
|
|
|
// flush disc ID and Devolution config out to memory
|
|
|
|
DCFlushRange(lowmem, 64);
|
|
|
|
}
|
|
|
|
|
2012-07-12 18:02:51 +02:00
|
|
|
void DEVO_Boot()
|
2012-07-12 17:53:04 +02:00
|
|
|
{
|
2012-07-12 18:02:51 +02:00
|
|
|
// the Devolution blob has an ID string at offset 4
|
|
|
|
puts((const char*)loader_bin + 4);
|
|
|
|
LAUNCH();
|
2012-07-12 17:53:04 +02:00
|
|
|
}
|
|
|
|
|
2012-07-12 15:21:14 +02:00
|
|
|
|
|
|
|
// General
|
2012-02-02 23:22:46 +01:00
|
|
|
#define SRAM_ENGLISH 0
|
|
|
|
#define SRAM_GERMAN 1
|
|
|
|
#define SRAM_FRENCH 2
|
|
|
|
#define SRAM_SPANISH 3
|
|
|
|
#define SRAM_ITALIAN 4
|
|
|
|
#define SRAM_DUTCH 5
|
|
|
|
|
2012-01-21 22:15:45 +01:00
|
|
|
syssram* __SYS_LockSram();
|
|
|
|
u32 __SYS_UnlockSram(u32 write);
|
|
|
|
u32 __SYS_SyncSram(void);
|
|
|
|
|
2012-07-12 17:53:04 +02:00
|
|
|
void GC_SetVideoMode(u8 videomode, u8 videoSetting)
|
2012-01-21 22:15:45 +01:00
|
|
|
{
|
2012-06-24 20:11:12 +02:00
|
|
|
syssram *sram;
|
|
|
|
sram = __SYS_LockSram();
|
|
|
|
static GXRModeObj *rmode;
|
|
|
|
int memflag = 0;
|
2012-02-21 18:21:12 +01:00
|
|
|
|
2012-06-14 17:27:57 +02:00
|
|
|
if((VIDEO_HaveComponentCable() && (CONF_GetProgressiveScan() > 0)) || videomode > 3)
|
2012-02-21 18:21:12 +01:00
|
|
|
sram->flags |= 0x80; //set progressive flag
|
|
|
|
else
|
|
|
|
sram->flags &= 0x7F; //clear progressive flag
|
|
|
|
|
2012-03-25 21:52:35 +02:00
|
|
|
if(videomode == 1 || videomode == 3 || videomode == 5)
|
2012-01-23 16:57:30 +01:00
|
|
|
{
|
2012-06-24 20:11:12 +02:00
|
|
|
memflag = 1;
|
2012-03-25 21:52:35 +02:00
|
|
|
sram->flags |= 0x01; // Set bit 0 to set the video mode to PAL
|
|
|
|
sram->ntd |= 0x40; //set pal60 flag
|
2012-01-23 16:57:30 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-03-25 21:52:35 +02:00
|
|
|
sram->flags &= 0xFE; // Clear bit 0 to set the video mode to NTSC
|
|
|
|
sram->ntd &= 0xBF; //clear pal60 flag
|
|
|
|
}
|
|
|
|
|
|
|
|
if(videomode == 1)
|
2012-06-24 20:11:12 +02:00
|
|
|
{
|
2012-07-12 17:53:04 +02:00
|
|
|
if(DMLCfg != NULL && videoSetting == 2)
|
2012-06-24 20:11:12 +02:00
|
|
|
DMLCfg->VideoMode |= DML_VID_FORCE_PAL50;
|
|
|
|
rmode = &TVPal528IntDf;
|
|
|
|
}
|
2012-03-25 21:52:35 +02:00
|
|
|
else if(videomode == 2)
|
2012-06-24 20:11:12 +02:00
|
|
|
{
|
2012-07-12 17:53:04 +02:00
|
|
|
if(DMLCfg != NULL && videoSetting == 2)
|
2012-06-24 20:11:12 +02:00
|
|
|
DMLCfg->VideoMode |= DML_VID_FORCE_NTSC;
|
|
|
|
rmode = &TVNtsc480IntDf;
|
|
|
|
}
|
2012-03-25 21:52:35 +02:00
|
|
|
else if(videomode == 3)
|
|
|
|
{
|
2012-07-12 17:53:04 +02:00
|
|
|
if(DMLCfg != NULL && videoSetting == 2)
|
2012-06-24 20:11:12 +02:00
|
|
|
DMLCfg->VideoMode |= DML_VID_FORCE_PAL60;
|
|
|
|
rmode = &TVEurgb60Hz480IntDf;
|
|
|
|
memflag = 5;
|
|
|
|
}
|
|
|
|
else if(videomode == 4 ||videomode == 6)
|
|
|
|
{
|
2012-07-12 17:53:04 +02:00
|
|
|
if(DMLCfg != NULL && videoSetting == 2)
|
2012-06-24 20:11:12 +02:00
|
|
|
DMLCfg->VideoMode |= DML_VID_FORCE_PROG;
|
|
|
|
rmode = &TVNtsc480Prog;
|
2012-03-25 21:52:35 +02:00
|
|
|
}
|
2012-06-24 20:11:12 +02:00
|
|
|
else if(videomode == 5 || videomode == 7)
|
2012-03-25 21:52:35 +02:00
|
|
|
{
|
2012-07-12 17:53:04 +02:00
|
|
|
if(DMLCfg != NULL && videoSetting == 2)
|
2012-06-24 20:11:12 +02:00
|
|
|
DMLCfg->VideoMode |= DML_VID_FORCE_PROG;
|
|
|
|
rmode = &TVNtsc480Prog;
|
|
|
|
memflag = 5;
|
2012-01-23 16:57:30 +01:00
|
|
|
}
|
2012-02-21 18:01:57 +01:00
|
|
|
|
2012-01-21 22:15:45 +01:00
|
|
|
__SYS_UnlockSram(1); // 1 -> write changes
|
|
|
|
while(!__SYS_SyncSram());
|
|
|
|
|
2012-06-14 17:27:57 +02:00
|
|
|
/* Set video mode register */
|
2012-06-24 20:11:12 +02:00
|
|
|
*(vu32 *)0x800000CC = memflag;
|
2012-03-11 20:47:24 +01:00
|
|
|
DCFlushRange((void *)(0x800000CC), 4);
|
2012-03-25 21:52:35 +02:00
|
|
|
|
2012-06-13 22:30:35 +02:00
|
|
|
/* Set video mode */
|
2012-06-24 20:11:12 +02:00
|
|
|
if (rmode != 0)
|
|
|
|
VIDEO_Configure(rmode);
|
2012-03-25 21:52:35 +02:00
|
|
|
|
2012-06-13 22:30:35 +02:00
|
|
|
/* Setup video */
|
2012-06-14 17:27:57 +02:00
|
|
|
VIDEO_SetBlack(TRUE);
|
|
|
|
VIDEO_Flush();
|
|
|
|
VIDEO_WaitVSync();
|
2012-06-24 20:11:12 +02:00
|
|
|
if(rmode->viTVMode & VI_NON_INTERLACE)
|
2012-06-14 17:27:57 +02:00
|
|
|
VIDEO_WaitVSync();
|
2012-02-02 23:22:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
u8 get_wii_language()
|
|
|
|
{
|
|
|
|
switch (CONF_GetLanguage())
|
|
|
|
{
|
|
|
|
case CONF_LANG_GERMAN:
|
|
|
|
return SRAM_GERMAN;
|
|
|
|
case CONF_LANG_FRENCH:
|
|
|
|
return SRAM_FRENCH;
|
|
|
|
case CONF_LANG_SPANISH:
|
|
|
|
return SRAM_SPANISH;
|
|
|
|
case CONF_LANG_ITALIAN:
|
|
|
|
return SRAM_ITALIAN;
|
|
|
|
case CONF_LANG_DUTCH:
|
|
|
|
return SRAM_DUTCH;
|
|
|
|
default:
|
|
|
|
return SRAM_ENGLISH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-11 20:47:24 +01:00
|
|
|
void GC_SetLanguage(u8 lang)
|
2012-02-02 23:22:46 +01:00
|
|
|
{
|
|
|
|
if (lang == 0)
|
|
|
|
lang = get_wii_language();
|
|
|
|
else
|
|
|
|
lang--;
|
|
|
|
|
|
|
|
syssram *sram;
|
|
|
|
sram = __SYS_LockSram();
|
|
|
|
sram->lang = lang;
|
|
|
|
|
|
|
|
__SYS_UnlockSram(1); // 1 -> write changes
|
|
|
|
while(!__SYS_SyncSram());
|
2012-02-25 23:34:52 +01:00
|
|
|
}
|
|
|
|
|
2012-04-24 21:42:18 +02:00
|
|
|
int GC_GameIsInstalled(char *discid, const char* partition, const char* dmlgamedir)
|
2012-02-25 23:34:52 +01:00
|
|
|
{
|
2012-03-06 22:04:23 +01:00
|
|
|
char folder[50];
|
2012-03-05 10:48:13 +01:00
|
|
|
char source[300];
|
2012-03-06 22:04:23 +01:00
|
|
|
snprintf(folder, sizeof(folder), dmlgamedir, partition);
|
2012-03-05 11:01:07 +01:00
|
|
|
snprintf(source, sizeof(source), "%s/%s/game.iso", folder, discid);
|
2012-02-25 23:34:52 +01:00
|
|
|
|
2012-04-22 15:39:26 +02:00
|
|
|
FILE *f = fopen(source, "rb");
|
|
|
|
if(f)
|
2012-02-25 23:34:52 +01:00
|
|
|
{
|
2012-03-05 10:48:13 +01:00
|
|
|
gprintf("Found on %s: %s\n", partition, source);
|
2012-02-25 23:34:52 +01:00
|
|
|
fclose(f);
|
2012-04-24 21:42:18 +02:00
|
|
|
return 1;
|
2012-02-25 23:34:52 +01:00
|
|
|
}
|
2012-04-22 15:39:26 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
snprintf(source, sizeof(source), "%s/%s/sys/boot.bin", folder, discid);
|
|
|
|
f = fopen(source, "rb");
|
|
|
|
if(f)
|
|
|
|
{
|
|
|
|
gprintf("Found on %s: %s\n", partition, source);
|
|
|
|
fclose(f);
|
2012-04-24 21:42:18 +02:00
|
|
|
return 2;
|
2012-04-22 15:39:26 +02:00
|
|
|
}
|
|
|
|
}
|
2012-04-24 21:42:18 +02:00
|
|
|
return 0;
|
2012-02-25 23:34:52 +01:00
|
|
|
}
|