usbloadergx/source/homebrewboot/BootHomebrew.cpp

171 lines
3.2 KiB
C++
Raw Normal View History

#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>
#include "Controls/DeviceHandler.hpp"
#include "../lstub.h"
#include "../sys.h"
#include "../gecko.h"
#include "dolloader.h"
2010-09-24 02:48:03 +02:00
static u8 *homebrewbuffer = (u8 *) 0x92000000;
static u32 homebrewsize = 0;
static std::vector<std::string> Arguments;
2010-09-24 02:48:03 +02:00
extern const u8 app_booter_dol[];
extern const u32 app_booter_dol_size;
2010-09-24 02:48:03 +02:00
void AddBootArgument(const char * argv)
{
2010-09-24 02:48:03 +02:00
std::string arg(argv);
Arguments.push_back(arg);
}
2010-09-24 02:48:03 +02:00
int CopyHomebrewMemory(u8 *temp, u32 pos, u32 len)
{
homebrewsize += len;
2010-09-24 02:48:03 +02:00
memcpy((homebrewbuffer) + pos, temp, len);
return 1;
}
void FreeHomebrewBuffer()
{
2010-09-24 02:48:03 +02:00
homebrewbuffer = (u8 *) 0x92000000;
homebrewsize = 0;
Arguments.clear();
}
2010-09-24 02:48:03 +02:00
static int SetupARGV(struct __argv * args)
{
2010-09-24 02:48:03 +02:00
if (!args) return -1;
2010-09-24 02:48:03 +02:00
bzero(args, sizeof(struct __argv));
args->argvMagic = ARGV_MAGIC;
u32 stringlength = 1;
/** Append Arguments **/
2010-09-24 02:48:03 +02:00
for (u32 i = 0; i < Arguments.size(); i++)
{
stringlength += Arguments[i].size() + 1;
}
args->length = stringlength;
2010-09-24 02:48:03 +02:00
args->commandLine = (char*) malloc(args->length);
2010-09-24 02:48:03 +02:00
if (!args->commandLine) return -1;
u32 argc = 0;
u32 position = 0;
/** Append Arguments **/
2010-09-24 02:48:03 +02:00
for (u32 i = 0; i < Arguments.size(); i++)
{
2010-09-24 02:48:03 +02:00
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 int RunAppbooter()
{
2010-09-24 02:48:03 +02:00
if (homebrewsize == 0) return -1;
ExitApp();
struct __argv args;
2010-09-24 02:48:03 +02:00
SetupARGV(&args);
u32 cpu_isr;
2010-09-24 02:48:03 +02:00
entrypoint entry = (entrypoint) load_dol((void*) app_booter_dol, &args);
2010-09-24 02:48:03 +02:00
if (!entry)
{
FreeHomebrewBuffer();
return -1;
}
u64 currentStub = getStubDest();
loadStub();
2010-09-24 02:48:03 +02:00
if (Set_Stub_Split(0x00010001, "UNEO") < 0)
{
2010-09-24 02:48:03 +02:00
if (Set_Stub_Split(0x00010001, "ULNR") < 0)
{
2010-09-24 02:48:03 +02:00
if (!currentStub) currentStub = 0x100000002ULL;
2010-09-24 02:48:03 +02:00
Set_Stub(currentStub);
}
}
2010-09-24 02:48:03 +02:00
SYS_ResetSystem(SYS_SHUTDOWN, 0, 0);
_CPU_ISR_Disable( cpu_isr );
__exception_closeall();
entry();
_CPU_ISR_Restore( cpu_isr );
return 0;
}
int BootHomebrew(const char * filepath)
{
void *buffer = NULL;
u32 filesize = 0;
2010-09-24 02:48:03 +02:00
FILE *file = fopen(filepath, "rb");
if (!file) return -1;
2010-09-24 02:48:03 +02:00
fseek(file, 0, SEEK_END);
filesize = ftell(file);
rewind(file);
2010-09-24 02:48:03 +02:00
buffer = malloc(filesize);
2010-09-24 02:48:03 +02:00
if (fread(buffer, 1, filesize, file) != filesize)
{
2010-09-24 02:48:03 +02:00
fclose(file);
free(buffer);
DeviceHandler::DestroyInstance();
Sys_BackToLoader();
}
2010-09-24 02:48:03 +02:00
fclose(file);
2010-09-24 02:48:03 +02:00
CopyHomebrewMemory((u8*) buffer, 0, filesize);
2010-09-24 02:48:03 +02:00
if (buffer)
{
2010-09-24 02:48:03 +02:00
free(buffer);
buffer = NULL;
}
2010-09-24 02:48:03 +02:00
AddBootArgument(filepath);
return RunAppbooter();
}
int BootHomebrewFromMem()
{
return RunAppbooter();
}