WiiFlow_Lite/source/homebrew/homebrew.cpp

150 lines
3.1 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 <ogc/lwp_threads.h>
#include <vector>
2012-01-21 21:57:41 +01:00
#include <string>
#include "homebrew.h"
2013-01-06 19:33:29 +01:00
#include "banner/AnimatedBanner.h"
#include "gecko/gecko.hpp"
2012-01-21 21:57:41 +01:00
#define EXECUTE_ADDR ((u8 *)0x92000000)
#define BOOTER_ADDR ((u8 *)0x93100000)
#define ARGS_ADDR ((u8 *)0x93200000)
#define BOOTER_ENTRY ((entry)BOOTER_ADDR)
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;
u8 valid = 0;
static vector<string> Arguments;
2012-01-21 21:57:41 +01:00
2012-08-17 20:28:39 +02:00
static bool IsDollZ(u8 *buf)
{
2012-08-17 20:28:39 +02:00
u8 cmp1[] = {0x3C};
return memcmp(&buf[0x100], cmp1, sizeof(cmp1)) == 0;
}
2012-08-17 20:28:39 +02:00
static bool IsSpecialELF(u8 *buf)
{
u32 cmp1[] = {0x7F454C46};
u8 cmp2[] = {0x00};
return memcmp(buf, cmp1, sizeof(cmp1)) == 0 && memcmp(&buf[0x24], cmp2, sizeof(cmp2)) == 0;
}
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 == NULL)
return -1;
2012-01-21 21:57:41 +01:00
FILE *file = fopen(filepath, "rb");
if(file == NULL)
return -2;
2012-01-21 21:57:41 +01:00
fseek(file, 0, SEEK_END);
u32 filesize = ftell(file);
if(filesize <= ((u32)BOOTER_ADDR - (u32)EXECUTE_ADDR))
{
rewind(file);
valid = (fread(EXECUTE_ADDR, 1, filesize, file) == filesize);
DCFlushRange(EXECUTE_ADDR, filesize);
}
fclose(file);
2012-01-21 21:57:41 +01:00
return valid;
2012-01-21 21:57:41 +01:00
}
int SetupARGV(struct __argv * args)
2012-01-21 21:57:41 +01:00
{
if(!args)
return -1;
2012-01-21 21:57:41 +01:00
memset(args, 0, sizeof(struct __argv));
2012-01-21 21:57:41 +01:00
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;
}
void writeStub()
{
/* Clear potential homebrew channel stub */
memset((void*)0x80001800, 0, 0x1800);
2013-01-06 19:33:29 +01:00
/* Extract our stub */
u32 StubSize = 0;
u8 *Stub = DecompressCopy(stub_bin, stub_bin_size, &StubSize);
/* Copy our own stub into memory */
2013-01-06 19:33:29 +01:00
memcpy((void*)0x80001800, Stub, StubSize);
DCFlushRange((void*)0x80001800, StubSize);
/* And free the memory again */
free(Stub);
}
void BootHomebrew()
2012-01-21 21:57:41 +01:00
{
__argv args;
2012-08-17 20:28:39 +02:00
if(!IsDollZ(EXECUTE_ADDR) && !IsSpecialELF(EXECUTE_ADDR))
SetupARGV(&args);
2012-08-17 20:28:39 +02:00
else
gprintf("Homebrew Boot Arguments disabled\n");
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);
memmove(ARGS_ADDR, &args, sizeof(args));
DCFlushRange(ARGS_ADDR, sizeof(args) + args.length);
JumpToEntry(BOOTER_ENTRY);
}
void JumpToEntry(entry EntryPoint)
{
gprintf("Jumping to %08x\n", EntryPoint);
SYS_ResetSystem(SYS_SHUTDOWN, 0, 0);
__lwp_thread_stopmultitasking(EntryPoint);
}