WiiFlow_Lite/source/homebrew/homebrew.cpp

164 lines
3.5 KiB
C++
Raw Normal View History

2012-01-21 21:57:41 +01:00
#include <gccore.h>
#include <ogcsys.h>
#include <stdio.h>
#include <string.h>
#include <ogc/machine/processor.h>
#include <vector>
2012-01-21 21:57:41 +01:00
#include <string>
#include "smartptr.hpp"
#include "gecko.h"
#include "mem2.hpp"
2012-01-21 21:57:41 +01:00
#define EXECUTE_ADDR ((u8 *)0x92000000)
#define BOOTER_ADDR ((u8 *)0x93000000)
#define ARGS_ADDR ((u8 *)0x93200000)
2012-01-21 21:57:41 +01:00
using namespace std;
2012-01-21 21:57:41 +01:00
extern const u8 app_booter_bin[];
extern const u32 app_booter_bin_size;
extern const u8 stub_bin[];
extern const u32 stub_bin_size;
2012-01-21 21:57:41 +01:00
typedef void (*entrypoint) (void);
extern "C" { void __exception_closeall(); }
u32 buffer_size = 0;
static vector<string> Arguments;
2012-01-21 21:57:41 +01:00
static u32 stubtitlepositions[8] = { 0x80001bf2, 0x80001bf3, 0x80001c06, 0x80001c07,
0x80001bfa, 0x80001bfb, 0x80001c0a, 0x80001c0b };
bool IsDollZ (u8 *buff)
{
u8 dollz_stamp[] = {0x3C};
int dollz_offs = 0x100;
int ret = memcmp (&buff[dollz_offs], dollz_stamp, sizeof(dollz_stamp));
if (ret == 0)
return true;
return false;
}
2012-01-21 21:57:41 +01:00
void AddBootArgument(const char * argv)
{
string arg(argv);
2012-01-21 21:57:41 +01:00
Arguments.push_back(arg);
}
int LoadHomebrew(const char *filepath)
2012-01-21 21:57:41 +01:00
{
if(!filepath)
return -1;
2012-01-21 21:57:41 +01:00
FILE *file = fopen(filepath ,"rb");
if(!file)
return -2;
2012-01-21 21:57:41 +01:00
fseek(file, 0, SEEK_END);
u32 filesize = ftell(file);
rewind(file);
buffer_size = filesize;
fread(EXECUTE_ADDR, 1, buffer_size, file);
DCFlushRange(EXECUTE_ADDR, buffer_size);
fclose(file);
2012-01-21 21:57:41 +01:00
return 1;
2012-01-21 21:57:41 +01:00
}
static int SetupARGV(struct __argv * args)
{
if(!args)
return -1;
2012-01-21 21:57:41 +01:00
bzero(args, sizeof(struct __argv));
args->argvMagic = ARGV_MAGIC;
u32 argc = 0;
u32 position = 0;
u32 stringlength = 1;
/** Append Arguments **/
for(u32 i = 0; i < Arguments.size(); i++)
stringlength += Arguments[i].size()+1;
args->length = stringlength;
//! Put the argument into mem2 too, to avoid overwriting it
args->commandLine = (char *) ARGS_ADDR + sizeof(struct __argv);
/** Append Arguments **/
for(u32 i = 0; i < Arguments.size(); i++)
{
strcpy(&args->commandLine[position], Arguments[i].c_str());
position += Arguments[i].size() + 1;
argc++;
}
args->argc = argc;
args->commandLine[args->length - 1] = '\0';
args->argv = &args->commandLine;
args->endARGV = args->argv + 1;
Arguments.clear();
return 0;
}
static void writeStub(u64 chan_title)
{
u8 i;
u32 digit;
char title[2][9];
snprintf(title[0], sizeof(title[0]), "%08x", TITLE_UPPER(chan_title));
snprintf(title[1], sizeof(title[1]), "%08x", TITLE_LOWER(chan_title));
/* Clear potential homebrew channel stub */
memset((void*)0x80001800, 0, 0x1800);
/* Copy our own stub into memory */
memcpy((void*)0x80001800, stub_bin, stub_bin_size);
/* Write in the Title ID we got */
for(i = 0; i < 4; i++)
{
sscanf(&title[0][i*2], "%02x", &digit);
//gprintf("%x\n", digit);
*(vu8*)stubtitlepositions[i] = digit;
sscanf(&title[1][i*2], "%02x", &digit);
//gprintf("%x\n", digit);
*(vu8*)stubtitlepositions[i+4] = digit;
}
DCFlushRange((void*)0x80001800, stub_bin_size);
}
int BootHomebrew(u64 chan_title)
2012-01-21 21:57:41 +01:00
{
writeStub(chan_title);
2012-01-21 21:57:41 +01:00
struct __argv args;
if (!IsDollZ(EXECUTE_ADDR))
SetupARGV(&args);
2012-01-21 21:57:41 +01:00
memcpy(BOOTER_ADDR, app_booter_bin, app_booter_bin_size);
DCFlushRange(BOOTER_ADDR, app_booter_bin_size);
entrypoint exeEntryPoint = (entrypoint)BOOTER_ADDR;
u32 cookie;
memmove(ARGS_ADDR, &args, sizeof(args));
DCFlushRange(ARGS_ADDR, sizeof(args) + args.length);
/* cleaning up and load dol */
SYS_ResetSystem(SYS_SHUTDOWN, 0, 0);
_CPU_ISR_Disable(cookie);
__exception_closeall();
exeEntryPoint();
_CPU_ISR_Restore(cookie);
2012-01-21 21:57:41 +01:00
return 0;
}