2010-09-19 01:04:39 +02:00
|
|
|
#include <gccore.h>
|
|
|
|
#include <ogcsys.h>
|
|
|
|
#include <malloc.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <ogc/machine/processor.h>
|
|
|
|
#include <wiiuse/wpad.h>
|
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
2010-12-31 00:49:22 +01:00
|
|
|
#include "Controls/DeviceHandler.hpp"
|
2011-07-23 21:49:51 +02:00
|
|
|
#include "settings/CSettings.h"
|
|
|
|
#include "lstub.h"
|
|
|
|
#include "sys.h"
|
|
|
|
#include "gecko.h"
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2011-07-23 21:49:51 +02:00
|
|
|
#define EXECUTE_ADDR ((u8 *) 0x92000000)
|
|
|
|
#define BOOTER_ADDR ((u8 *) 0x93000000)
|
|
|
|
#define ARGS_ADDR ((u8 *) 0x93200000)
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2011-07-23 21:49:51 +02:00
|
|
|
extern const u8 app_booter_bin[];
|
|
|
|
extern const u32 app_booter_bin_size;
|
|
|
|
|
|
|
|
typedef void (*entrypoint) (void);
|
|
|
|
extern "C" { void __exception_closeall(); }
|
|
|
|
|
|
|
|
static u8 *homebrewbuffer = EXECUTE_ADDR;
|
2010-11-06 16:30:14 +01:00
|
|
|
static u32 homebrewsize = 0;
|
2010-09-19 01:04:39 +02:00
|
|
|
static std::vector<std::string> Arguments;
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
void AddBootArgument(const char * argv)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
std::string arg(argv);
|
|
|
|
Arguments.push_back(arg);
|
2010-09-19 01:04:39 +02:00
|
|
|
}
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
int CopyHomebrewMemory(u8 *temp, u32 pos, u32 len)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
|
|
|
homebrewsize += len;
|
2010-09-24 02:48:03 +02:00
|
|
|
memcpy((homebrewbuffer) + pos, temp, len);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FreeHomebrewBuffer()
|
|
|
|
{
|
2011-07-23 21:49:51 +02:00
|
|
|
homebrewbuffer = EXECUTE_ADDR;
|
2010-09-19 01:04:39 +02:00
|
|
|
homebrewsize = 0;
|
|
|
|
|
|
|
|
Arguments.clear();
|
|
|
|
}
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
static int SetupARGV(struct __argv * args)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
if (!args) return -1;
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
bzero(args, sizeof(struct __argv));
|
2010-09-19 01:04:39 +02:00
|
|
|
args->argvMagic = ARGV_MAGIC;
|
|
|
|
|
2011-07-25 19:50:03 +02:00
|
|
|
u32 argc = 0;
|
|
|
|
u32 position = 0;
|
2010-09-19 01:04:39 +02:00
|
|
|
u32 stringlength = 1;
|
|
|
|
|
|
|
|
/** Append Arguments **/
|
2010-09-24 02:48:03 +02:00
|
|
|
for (u32 i = 0; i < Arguments.size(); i++)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
2010-09-19 01:16:05 +02:00
|
|
|
stringlength += Arguments[i].size() + 1;
|
2010-09-19 01:04:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
args->length = stringlength;
|
2011-07-25 19:50:03 +02:00
|
|
|
//! Put the argument into mem2 too, to avoid overwriting it
|
|
|
|
args.commandLine = (char *) ARGS_ADDR + sizeof(struct __argv);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
|
|
|
/** Append Arguments **/
|
2010-09-24 02:48:03 +02:00
|
|
|
for (u32 i = 0; i < Arguments.size(); i++)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
strcpy(&args->commandLine[position], Arguments[i].c_str());
|
2010-09-19 01:16:05 +02:00
|
|
|
position += Arguments[i].size() + 1;
|
|
|
|
argc++;
|
2010-09-19 01:04:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
args->argc = argc;
|
|
|
|
|
|
|
|
args->commandLine[args->length - 1] = '\0';
|
|
|
|
args->argv = &args->commandLine;
|
|
|
|
args->endARGV = args->argv + 1;
|
|
|
|
|
|
|
|
Arguments.clear();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int RunAppbooter()
|
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
if (homebrewsize == 0) return -1;
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-11-06 16:30:14 +01:00
|
|
|
ExitApp();
|
|
|
|
|
2011-07-23 21:49:51 +02:00
|
|
|
if(Settings.EntryIOS != IOS_GetVersion())
|
|
|
|
IOS_ReloadIOS(Settings.EntryIOS);
|
|
|
|
|
2010-09-19 01:04:39 +02:00
|
|
|
struct __argv args;
|
2010-09-24 02:48:03 +02:00
|
|
|
SetupARGV(&args);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
|
|
|
u32 cpu_isr;
|
|
|
|
|
2011-07-23 21:49:51 +02:00
|
|
|
memcpy(BOOTER_ADDR, app_booter_bin, app_booter_bin_size);
|
|
|
|
DCFlushRange(BOOTER_ADDR, app_booter_bin_size);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2011-07-23 21:49:51 +02:00
|
|
|
entrypoint entry = (entrypoint) BOOTER_ADDR;
|
|
|
|
|
|
|
|
if (args.argvMagic == ARGV_MAGIC)
|
|
|
|
{
|
|
|
|
memmove(ARGS_ADDR, &args, sizeof(args));
|
2011-07-25 19:50:03 +02:00
|
|
|
DCFlushRange(ARGS_ADDR, sizeof(args) + args.length);
|
2011-07-23 21:49:51 +02:00
|
|
|
}
|
2010-09-19 01:04:39 +02:00
|
|
|
|
|
|
|
u64 currentStub = getStubDest();
|
|
|
|
loadStub();
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
if (Set_Stub_Split(0x00010001, "UNEO") < 0)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
if (Set_Stub_Split(0x00010001, "ULNR") < 0)
|
2010-09-19 01:16:05 +02:00
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
if (!currentStub) currentStub = 0x100000002ULL;
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
Set_Stub(currentStub);
|
2010-09-19 01:16:05 +02:00
|
|
|
}
|
2010-09-19 01:04:39 +02:00
|
|
|
}
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
SYS_ResetSystem(SYS_SHUTDOWN, 0, 0);
|
2010-09-19 01:04:39 +02:00
|
|
|
_CPU_ISR_Disable( cpu_isr );
|
|
|
|
__exception_closeall();
|
|
|
|
entry();
|
|
|
|
_CPU_ISR_Restore( cpu_isr );
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-11-06 16:30:14 +01:00
|
|
|
int BootHomebrew(const char * filepath)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
|
|
|
void *buffer = NULL;
|
|
|
|
u32 filesize = 0;
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
FILE *file = fopen(filepath, "rb");
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2011-01-22 17:10:36 +01:00
|
|
|
if (!file) return -1;
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
fseek(file, 0, SEEK_END);
|
|
|
|
filesize = ftell(file);
|
|
|
|
rewind(file);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
buffer = malloc(filesize);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
if (fread(buffer, 1, filesize, file) != filesize)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
fclose(file);
|
|
|
|
free(buffer);
|
2010-12-31 00:49:22 +01:00
|
|
|
DeviceHandler::DestroyInstance();
|
2010-09-19 01:16:05 +02:00
|
|
|
Sys_BackToLoader();
|
2010-09-19 01:04:39 +02:00
|
|
|
}
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
fclose(file);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
CopyHomebrewMemory((u8*) buffer, 0, filesize);
|
2010-09-19 01:04:39 +02:00
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
if (buffer)
|
2010-09-19 01:04:39 +02:00
|
|
|
{
|
2010-09-24 02:48:03 +02:00
|
|
|
free(buffer);
|
2010-09-19 01:16:05 +02:00
|
|
|
buffer = NULL;
|
2010-09-19 01:04:39 +02:00
|
|
|
}
|
|
|
|
|
2010-09-24 02:48:03 +02:00
|
|
|
AddBootArgument(filepath);
|
2010-09-19 01:04:39 +02:00
|
|
|
return RunAppbooter();
|
|
|
|
}
|
|
|
|
|
|
|
|
int BootHomebrewFromMem()
|
|
|
|
{
|
|
|
|
return RunAppbooter();
|
|
|
|
}
|