mirror of
https://github.com/Fledge68/WiiFlow_Lite.git
synced 2025-01-27 02:55:26 +01:00
1480483963
-added yet another security check to prevent crashes -added a new SOURCEFLOW option, remember_last_item (enabled by default), if disabled and you use another sourceflow, wiiflow will not jump back to the previously selected flow but to the first one again
520 lines
13 KiB
C++
520 lines
13 KiB
C++
/****************************************************************************
|
|
* Copyright (C) 2012 FIX94
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
****************************************************************************/
|
|
#include <gccore.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <ogc/machine/processor.h>
|
|
#include <ogc/lwp_threads.h>
|
|
// for directory parsing and low-level file I/O
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <dirent.h>
|
|
|
|
#include "gc/gc.hpp"
|
|
#include "gui/text.hpp"
|
|
#include "devicemounter/DeviceHandler.hpp"
|
|
#include "gecko/gecko.hpp"
|
|
#include "fileOps/fileOps.h"
|
|
#include "homebrew/homebrew.h"
|
|
#include "loader/utils.h"
|
|
#include "loader/disc.h"
|
|
#include "loader/sys.h"
|
|
#include "memory/memory.h"
|
|
#include "memory/mem2.hpp"
|
|
|
|
// DIOS-MIOS
|
|
DML_CFG DMLCfg;
|
|
|
|
void DML_New_SetOptions(const char *GamePath, char *CheatPath, const char *NewCheatPath,
|
|
const char *partition, bool cheats, bool debugger, u8 NMM, u8 nodisc, u8 DMLvideoMode,
|
|
u8 videoSetting, bool widescreen, bool new_dm_cfg, bool activity_led, bool screenshot)
|
|
{
|
|
gprintf("DIOS-MIOS: Launch game '%s' through memory (new method)\n", GamePath);
|
|
memset(&DMLCfg, 0, sizeof(DML_CFG));
|
|
|
|
DMLCfg.Magicbytes = 0xD1050CF6;
|
|
if(new_dm_cfg)
|
|
DMLCfg.CfgVersion = 0x00000002;
|
|
else
|
|
DMLCfg.CfgVersion = 0x00000001;
|
|
|
|
if(videoSetting == 0)
|
|
DMLCfg.VideoMode |= DML_VID_NONE;
|
|
else if(videoSetting == 1)
|
|
DMLCfg.VideoMode |= DML_VID_DML_AUTO;
|
|
else
|
|
DMLCfg.VideoMode |= DML_VID_FORCE;
|
|
|
|
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)
|
|
{
|
|
const char *ptr = NULL;
|
|
if(strncasecmp(CheatPath, partition, strlen(partition)) != 0)
|
|
{
|
|
fsop_CopyFile(CheatPath, NewCheatPath, NULL, NULL);
|
|
ptr = strchr(NewCheatPath, '/');
|
|
}
|
|
else
|
|
ptr = strchr(CheatPath, '/');
|
|
strncpy(DMLCfg.CheatPath, ptr, sizeof(DMLCfg.CheatPath));
|
|
gprintf("DIOS-MIOS: Cheat Path %s\n", ptr);
|
|
DMLCfg.Config |= DML_CFG_CHEAT_PATH;
|
|
}
|
|
if(screenshot)
|
|
DMLCfg.Config |= DML_CFG_SCREENSHOT;
|
|
if(activity_led)
|
|
DMLCfg.Config |= DML_CFG_ACTIVITY_LED;
|
|
if(cheats)
|
|
DMLCfg.Config |= DML_CFG_CHEATS;
|
|
if(debugger)
|
|
DMLCfg.Config |= DML_CFG_DEBUGGER;
|
|
if(NMM)
|
|
DMLCfg.Config |= DML_CFG_NMM;
|
|
if(NMM > 1)
|
|
DMLCfg.Config |= DML_CFG_NMM_DEBUG;
|
|
if(nodisc)
|
|
{
|
|
if(new_dm_cfg)
|
|
DMLCfg.Config |= DML_CFG_NODISC_CFG2;
|
|
else
|
|
DMLCfg.Config |= DML_CFG_NODISC_CFG1;
|
|
}
|
|
if(widescreen && new_dm_cfg)
|
|
DMLCfg.Config |= DML_CFG_FORCE_WIDE;
|
|
|
|
if(DMLvideoMode > 3)
|
|
DMLCfg.VideoMode |= DML_VID_PROG_PATCH;
|
|
}
|
|
|
|
void DML_Old_SetOptions(const char *GamePath)
|
|
{
|
|
gprintf("DIOS-MIOS: Launch game '%s' through boot.bin (old method)\n", GamePath);
|
|
fsop_WriteFile(DML_BOOT_PATH, GamePath, strlen(GamePath)+1);
|
|
|
|
//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(bool new_dm_cfg)
|
|
{
|
|
memset(&DMLCfg, 0, sizeof(DML_CFG));
|
|
|
|
DMLCfg.Magicbytes = 0xD1050CF6;
|
|
if(new_dm_cfg)
|
|
DMLCfg.CfgVersion = 0x00000002;
|
|
else
|
|
DMLCfg.CfgVersion = 0x00000001;
|
|
DMLCfg.VideoMode |= DML_VID_DML_AUTO;
|
|
|
|
DMLCfg.Config |= DML_CFG_BOOT_DISC;
|
|
}
|
|
|
|
void DML_New_WriteOptions()
|
|
{
|
|
//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));
|
|
}
|
|
|
|
|
|
// Nintendont
|
|
NIN_CFG NinCfg;
|
|
|
|
void Nintendont_SetOptions(const char *game, u8 NMM, u8 videoSetting, bool widescreen)
|
|
{
|
|
memset(&NinCfg, 0, sizeof(NIN_CFG));
|
|
NinCfg.Magicbytes = 0x01070CF6;
|
|
NinCfg.Version = 0x00000001;
|
|
NinCfg.Config |= NIN_CFG_AUTO_BOOT;
|
|
NinCfg.Config |= NIN_CFG_GAME_PATH; /* temporary */
|
|
|
|
if(videoSetting == 0)
|
|
NinCfg.VideoMode |= NIN_VID_NONE;
|
|
else if(videoSetting == 1)
|
|
NinCfg.VideoMode |= NIN_VID_AUTO;
|
|
else
|
|
NinCfg.VideoMode |= NIN_VID_FORCE;
|
|
|
|
if(widescreen)
|
|
NinCfg.Config |= NIN_CFG_FORCE_WIDE;
|
|
if(NMM > 0)
|
|
NinCfg.Config |= NIN_CFG_MEMCARDEMU;
|
|
|
|
strncpy(NinCfg.GamePath, strchr(game, '/') + 1, 254);
|
|
if(strstr(NinCfg.GamePath, "boot.bin") != NULL)
|
|
{
|
|
*strrchr(NinCfg.GamePath, '/') = '\0'; //boot.bin
|
|
*(strrchr(NinCfg.GamePath, '/')+1) = '\0'; //sys
|
|
}
|
|
gprintf("Nintendont Game Path: %s\n", NinCfg.GamePath);
|
|
}
|
|
|
|
void Nintendont_WriteOptions()
|
|
{
|
|
gprintf("Writing Nintendont CFG: %s\n", NIN_CFG_PATH);
|
|
fsop_WriteFile(NIN_CFG_PATH, &NinCfg, sizeof(NIN_CFG));
|
|
}
|
|
|
|
bool Nintendont_GetLoader()
|
|
{
|
|
bool ret = false;
|
|
for(u8 i = SD; i < MAXDEVICES; ++i)
|
|
{
|
|
const char *dol_path = fmt(NIN_LOADER_PATH, DeviceName[i]);
|
|
ret = (LoadHomebrew(dol_path) == 1);
|
|
if(ret == true)
|
|
{
|
|
gprintf("Nintendont loaded: %s\n", dol_path);
|
|
AddBootArgument(dol_path);
|
|
break;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// Devolution
|
|
u8 *tmp_buffer = NULL;
|
|
u8 *loader_bin = NULL;
|
|
u32 loader_size = 0;
|
|
extern "C" { extern void __exception_closeall(); }
|
|
static gconfig *DEVO_CONFIG = (gconfig*)0x80000020;
|
|
#define DEVO_ENTRY ((entry)loader_bin)
|
|
|
|
bool DEVO_Installed(const char *path)
|
|
{
|
|
loader_size = 0;
|
|
bool devo = false;
|
|
fsop_GetFileSizeBytes(fmt(DEVO_LOADER_PATH, path), &loader_size);
|
|
if(loader_size > 0x80) //Size should be more than 128b
|
|
{
|
|
gprintf("Devolution found\n");
|
|
devo = true;
|
|
}
|
|
return devo;
|
|
}
|
|
|
|
void DEVO_GetLoader(const char *path)
|
|
{
|
|
loader_size = 0;
|
|
tmp_buffer = fsop_ReadFile(fmt(DEVO_LOADER_PATH, path), &loader_size);
|
|
if(tmp_buffer == NULL)
|
|
gprintf("Devolution: Loader not found!\n");
|
|
}
|
|
|
|
void DEVO_SetOptions(const char *isopath, const char *gameID, bool memcard_emu,
|
|
bool widescreen, bool activity_led, bool wifi)
|
|
{
|
|
// re-mount device we need
|
|
DeviceHandle.MountDevolution();
|
|
|
|
//start writing cfg to mem
|
|
struct stat st;
|
|
int data_fd;
|
|
char iso2path[256];
|
|
iso2path[255] = '\0';
|
|
|
|
stat(isopath, &st);
|
|
FILE *f = fopen(isopath, "rb");
|
|
gprintf("Devolution: ISO Header %s\n", isopath);
|
|
fread((u8*)Disc_ID, 1, 32, f);
|
|
fclose(f);
|
|
f = NULL;
|
|
|
|
// fill out the Devolution config struct
|
|
memset(DEVO_CONFIG, 0, sizeof(gconfig));
|
|
DEVO_CONFIG->signature = DEVO_CONFIG_SIG;
|
|
DEVO_CONFIG->version = DEVO_CONFIG_VERSION;
|
|
DEVO_CONFIG->device_signature = st.st_dev;
|
|
DEVO_CONFIG->disc1_cluster = st.st_ino;
|
|
|
|
// Pergame options
|
|
if(wifi)
|
|
DEVO_CONFIG->options |= DEVO_CONFIG_WIFILOG;
|
|
if(widescreen)
|
|
DEVO_CONFIG->options |= DEVO_CONFIG_WIDE;
|
|
if(!activity_led)
|
|
DEVO_CONFIG->options |= DEVO_CONFIG_NOLED;
|
|
|
|
// If 2nd iso file tell Devo about it
|
|
strncpy(iso2path, isopath, 255);
|
|
char *ptz = strstr(iso2path, "game.iso");
|
|
if(ptz != NULL)
|
|
{
|
|
strncpy(ptz, "gam1.iso", 8);
|
|
f = fopen(iso2path, "rb");
|
|
if(f == NULL)
|
|
{
|
|
strncpy(ptz, "gam2.iso", 8);
|
|
f = fopen(iso2path, "rb");
|
|
if(f == NULL)
|
|
{
|
|
strncpy(ptz, "disc2.iso", 9);
|
|
f = fopen(iso2path, "rb");
|
|
}
|
|
}
|
|
if(f != NULL)
|
|
{
|
|
gprintf("Devolution: 2nd ISO File for Multi DVD Game %s\n", iso2path);
|
|
stat(iso2path, &st);
|
|
DEVO_CONFIG->disc2_cluster = st.st_ino;
|
|
fclose(f);
|
|
}
|
|
}
|
|
|
|
// make sure these directories exist, they are required for Devolution to function correctly
|
|
fsop_MakeFolder(fmt("%s:/apps", DeviceName[currentPartition]));
|
|
fsop_MakeFolder(fmt("%s:/apps/gc_devo", DeviceName[currentPartition]));
|
|
|
|
if(memcard_emu)
|
|
{
|
|
const char *memcard_dir = NULL;
|
|
// 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 512KB (smallest possible memcard = 59 blocks)
|
|
if(gameID[3] == 'J') //Japanese Memory Card
|
|
memcard_dir = fmt("%s:/apps/gc_devo/memcard_jap.bin", DeviceName[currentPartition]);
|
|
else
|
|
memcard_dir = fmt("%s:/apps/gc_devo/memcard.bin", DeviceName[currentPartition]);
|
|
gprintf("Devolution: Memory Card File %s\n", memcard_dir);
|
|
// check if file doesn't exist
|
|
if(stat(memcard_dir, &st) == -1 || st.st_size < (1<<19))
|
|
{
|
|
// need to enlarge or create it
|
|
data_fd = open(memcard_dir, O_WRONLY|O_CREAT);
|
|
if(data_fd >= 0)
|
|
{
|
|
// try to make it 16MB (largest possible memcard = 2043 blocks)
|
|
gprintf("Devolution: Resizing Memory Card File...\n");
|
|
ftruncate(data_fd, 16<<20);
|
|
if(fstat(data_fd, &st) == -1 || st.st_size < (1<<19))
|
|
{
|
|
// 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;
|
|
}
|
|
}
|
|
gprintf("Devolution: Memory Card at %08x\n", st.st_ino);
|
|
}
|
|
else
|
|
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((void*)Disc_ID, 64);
|
|
|
|
DeviceHandle.UnMountDevolution();
|
|
}
|
|
|
|
void DEVO_Boot()
|
|
{
|
|
/* Move our loader into low MEM1 */
|
|
loader_bin = (u8*)MEM1_lo_alloc(loader_size);
|
|
memcpy(loader_bin, tmp_buffer, loader_size);
|
|
DCFlushRange(loader_bin, ALIGN32(loader_size));
|
|
MEM2_free(tmp_buffer);
|
|
gprintf("%s\n", (loader_bin+4));
|
|
/* Boot that binary */
|
|
JumpToEntry(DEVO_ENTRY);
|
|
}
|
|
|
|
|
|
// General
|
|
#define SRAM_ENGLISH 0
|
|
#define SRAM_GERMAN 1
|
|
#define SRAM_FRENCH 2
|
|
#define SRAM_SPANISH 3
|
|
#define SRAM_ITALIAN 4
|
|
#define SRAM_DUTCH 5
|
|
|
|
extern "C" {
|
|
syssram* __SYS_LockSram();
|
|
u32 __SYS_UnlockSram(u32 write);
|
|
u32 __SYS_SyncSram(void);
|
|
}
|
|
|
|
void GC_SetVideoMode(u8 videomode, u8 videoSetting, u8 loader)
|
|
{
|
|
syssram *sram;
|
|
sram = __SYS_LockSram();
|
|
GXRModeObj *vmode = VIDEO_GetPreferredMode(0);
|
|
int vmode_reg = 0;
|
|
|
|
if((VIDEO_HaveComponentCable() && (CONF_GetProgressiveScan() > 0)) || videomode > 3)
|
|
sram->flags |= 0x80; //set progressive flag
|
|
else
|
|
sram->flags &= 0x7F; //clear progressive flag
|
|
|
|
if(videomode == 1 || videomode == 3 || videomode == 5)
|
|
{
|
|
vmode_reg = 1;
|
|
sram->flags |= 0x01; // Set bit 0 to set the video mode to PAL
|
|
sram->ntd |= 0x40; //set pal60 flag
|
|
}
|
|
else
|
|
{
|
|
sram->flags &= 0xFE; // Clear bit 0 to set the video mode to NTSC
|
|
sram->ntd &= 0xBF; //clear pal60 flag
|
|
}
|
|
|
|
if(videomode == 1)
|
|
{
|
|
if(videoSetting == 2)
|
|
{
|
|
if(loader == 0)
|
|
DMLCfg.VideoMode |= DML_VID_FORCE_PAL50;
|
|
else if(loader == 2)
|
|
NinCfg.VideoMode |= NIN_VID_FORCE_PAL50;
|
|
}
|
|
vmode = &TVPal528IntDf;
|
|
}
|
|
else if(videomode == 2)
|
|
{
|
|
if(videoSetting == 2)
|
|
{
|
|
if(loader == 0)
|
|
DMLCfg.VideoMode |= DML_VID_FORCE_NTSC;
|
|
else if(loader == 2)
|
|
NinCfg.VideoMode |= NIN_VID_FORCE_NTSC;
|
|
}
|
|
vmode = &TVNtsc480IntDf;
|
|
}
|
|
else if(videomode == 3)
|
|
{
|
|
if(videoSetting == 2)
|
|
{
|
|
if(loader == 0)
|
|
DMLCfg.VideoMode |= DML_VID_FORCE_PAL60;
|
|
else if(loader == 2)
|
|
NinCfg.VideoMode |= NIN_VID_FORCE_PAL60;
|
|
}
|
|
vmode = &TVEurgb60Hz480IntDf;
|
|
vmode_reg = 5;
|
|
}
|
|
else if(videomode == 4 ||videomode == 6)
|
|
{
|
|
if(videoSetting == 2)
|
|
{
|
|
if(loader == 0)
|
|
DMLCfg.VideoMode |= DML_VID_FORCE_PROG;
|
|
else if(loader == 2)
|
|
NinCfg.Config |= NIN_CFG_FORCE_PROG;
|
|
}
|
|
vmode = &TVNtsc480Prog;
|
|
}
|
|
else if(videomode == 5 || videomode == 7)
|
|
{
|
|
if(videoSetting == 2)
|
|
{
|
|
if(loader == 0)
|
|
DMLCfg.VideoMode |= DML_VID_FORCE_PROG;
|
|
else if(loader == 2)
|
|
NinCfg.Config |= NIN_CFG_FORCE_PROG;
|
|
}
|
|
vmode = &TVNtsc480Prog;
|
|
vmode_reg = 5;
|
|
}
|
|
|
|
__SYS_UnlockSram(1); // 1 -> write changes
|
|
while(!__SYS_SyncSram());
|
|
|
|
/* Set video mode register */
|
|
*Video_Mode = vmode_reg;
|
|
DCFlushRange((void*)Video_Mode, 4);
|
|
|
|
/* Set video mode */
|
|
if(vmode != 0)
|
|
VIDEO_Configure(vmode);
|
|
|
|
/* Setup video */
|
|
VIDEO_SetBlack(FALSE);
|
|
VIDEO_Flush();
|
|
VIDEO_WaitVSync();
|
|
if(vmode->viTVMode & VI_NON_INTERLACE)
|
|
VIDEO_WaitVSync();
|
|
else while(VIDEO_GetNextField())
|
|
VIDEO_WaitVSync();
|
|
|
|
/* Set black and flush */
|
|
VIDEO_SetBlack(TRUE);
|
|
VIDEO_Flush();
|
|
VIDEO_WaitVSync();
|
|
if(vmode->viTVMode & VI_NON_INTERLACE)
|
|
VIDEO_WaitVSync();
|
|
else while(VIDEO_GetNextField())
|
|
VIDEO_WaitVSync();
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
void GC_SetLanguage(u8 lang)
|
|
{
|
|
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());
|
|
}
|