mirror of
https://github.com/wiidev/usbloadergx.git
synced 2025-01-23 00:41:10 +01:00
928020b32f
*Added Browser for AlternateDOL from disc feature NOTE: Right now the browser isn't working right and crashing from time to time so you i would not recommend to use it yet. It's also showing all files on the disc for test purpose. I'll try to fix it later right now i don't have time for it. *Changed some minor codes and moved cheats to its own folder
251 lines
5.0 KiB
C
251 lines
5.0 KiB
C
#include <ogcsys.h>
|
|
#include <stdio.h>
|
|
#include <gccore.h>
|
|
#include <string.h>
|
|
#include <malloc.h>
|
|
|
|
#include "fatmounter.h"
|
|
#include "apploader.h"
|
|
#include "wdvd.h"
|
|
#include "fstfile.h"
|
|
|
|
/** Alternate dolloader made by WiiPower modified by dimok **/
|
|
|
|
bool Load_Dol(void **buffer, int* dollen, char * filepath)
|
|
{
|
|
int ret;
|
|
FILE* file;
|
|
void* dol_buffer;
|
|
|
|
char fullpath[200];
|
|
char gameidbuffer6[7];
|
|
memset(gameidbuffer6, 0, 7);
|
|
memcpy(gameidbuffer6, (char*)0x80000000, 6);
|
|
snprintf(fullpath, 200, "%s%s.dol", filepath, gameidbuffer6);
|
|
|
|
SDCard_Init();
|
|
USBDevice_Init();
|
|
|
|
file = fopen(fullpath, "rb");
|
|
|
|
if(file == NULL)
|
|
{
|
|
fclose(file);
|
|
SDCard_deInit();
|
|
USBDevice_deInit();
|
|
return false;
|
|
}
|
|
|
|
int filesize;
|
|
fseek(file, 0, SEEK_END);
|
|
filesize = ftell(file);
|
|
fseek(file, 0, SEEK_SET);
|
|
|
|
dol_buffer = malloc(filesize);
|
|
if (dol_buffer == NULL)
|
|
{
|
|
fclose(file);
|
|
SDCard_deInit();
|
|
USBDevice_deInit();
|
|
return false;
|
|
}
|
|
ret = fread( dol_buffer, 1, filesize, file);
|
|
if(ret != filesize)
|
|
{
|
|
free(dol_buffer);
|
|
fclose(file);
|
|
SDCard_deInit();
|
|
USBDevice_deInit();
|
|
return false;
|
|
}
|
|
fclose(file);
|
|
|
|
SDCard_deInit();
|
|
USBDevice_deInit();
|
|
*buffer = dol_buffer;
|
|
*dollen = filesize;
|
|
return true;
|
|
}
|
|
|
|
bool Remove_001_Protection(void *Address, int Size)
|
|
{
|
|
u8 SearchPattern[16] = { 0x40, 0x82, 0x00, 0x0C, 0x38, 0x60, 0x00, 0x01, 0x48, 0x00, 0x02, 0x44, 0x38, 0x61, 0x00, 0x18 };
|
|
u8 PatchData[16] = { 0x40, 0x82, 0x00, 0x04, 0x38, 0x60, 0x00, 0x01, 0x48, 0x00, 0x02, 0x44, 0x38, 0x61, 0x00, 0x18 };
|
|
|
|
void *Addr = Address;
|
|
void *Addr_end = Address+Size;
|
|
|
|
while(Addr <= Addr_end-sizeof(SearchPattern))
|
|
{
|
|
if(memcmp(Addr, SearchPattern, sizeof(SearchPattern))==0)
|
|
{
|
|
memcpy(Addr,PatchData,sizeof(PatchData));
|
|
return true;
|
|
}
|
|
Addr += 4;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
typedef struct _dolheader {
|
|
u32 text_pos[7];
|
|
u32 data_pos[11];
|
|
u32 text_start[7];
|
|
u32 data_start[11];
|
|
u32 text_size[7];
|
|
u32 data_size[11];
|
|
u32 bss_start;
|
|
u32 bss_size;
|
|
u32 entry_point;
|
|
} dolheader;
|
|
|
|
static dolheader *dolfile;
|
|
|
|
|
|
u32 load_dol_image(void *dolstart) {
|
|
|
|
u32 i;
|
|
|
|
if (dolstart) {
|
|
dolfile = (dolheader *) dolstart;
|
|
for (i = 0; i < 7; i++) {
|
|
if ((!dolfile->text_size[i]) || (dolfile->text_start[i] < 0x100)) continue;
|
|
VIDEO_WaitVSync();
|
|
ICInvalidateRange ((void *) dolfile->text_start[i],dolfile->text_size[i]);
|
|
memmove ((void *) dolfile->text_start[i],dolstart+dolfile->text_pos[i],dolfile->text_size[i]);
|
|
}
|
|
|
|
for(i = 0; i < 11; i++) {
|
|
if ((!dolfile->data_size[i]) || (dolfile->data_start[i] < 0x100)) continue;
|
|
VIDEO_WaitVSync();
|
|
memmove ((void *) dolfile->data_start[i],dolstart+dolfile->data_pos[i],dolfile->data_size[i]);
|
|
DCFlushRangeNoSync ((void *) dolfile->data_start[i],dolfile->data_size[i]);
|
|
}
|
|
/*
|
|
memset ((void *) dolfile->bss_start, 0, dolfile->bss_size);
|
|
DCFlushRange((void *) dolfile->bss_start, dolfile->bss_size);
|
|
*/
|
|
return dolfile->entry_point;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int i;
|
|
static int phase;
|
|
|
|
u32 load_dol_start(void *dolstart)
|
|
{
|
|
if (dolstart)
|
|
{
|
|
dolfile = (dolheader *)dolstart;
|
|
return dolfile->entry_point;
|
|
} else
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
memset((void *)dolfile->bss_start, 0, dolfile->bss_size);
|
|
DCFlushRange((void *)dolfile->bss_start, dolfile->bss_size);
|
|
|
|
phase = 0;
|
|
i = 0;
|
|
}
|
|
|
|
bool load_dol_image_modified(void **offset, u32 *pos, u32 *len)
|
|
{
|
|
if (phase == 0)
|
|
{
|
|
if (i == 7)
|
|
{
|
|
phase = 1;
|
|
i = 0;
|
|
} else
|
|
{
|
|
if ((!dolfile->text_size[i]) || (dolfile->text_start[i] < 0x100))
|
|
{
|
|
*offset = 0;
|
|
*pos = 0;
|
|
*len = 0;
|
|
} else
|
|
{
|
|
*offset = (void *)dolfile->text_start[i];
|
|
*pos = dolfile->text_pos[i];
|
|
*len = dolfile->text_size[i];
|
|
}
|
|
i++;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (phase == 1)
|
|
{
|
|
if (i == 11)
|
|
{
|
|
phase = 2;
|
|
return false;
|
|
}
|
|
|
|
if ((!dolfile->data_size[i]) || (dolfile->data_start[i] < 0x100))
|
|
{
|
|
*offset = 0;
|
|
*pos = 0;
|
|
*len = 0;
|
|
} else
|
|
{
|
|
*offset = (void *)dolfile->data_start[i];
|
|
*pos = dolfile->data_pos[i];
|
|
*len = dolfile->data_size[i];
|
|
}
|
|
i++;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
u32 Load_Dol_from_disc(u32 doloffset, u8 videoSelected, u8 patchcountrystring, u8 vipatch)
|
|
{
|
|
int ret;
|
|
void *dol_header;
|
|
u32 entrypoint;
|
|
|
|
dol_header = memalign(32, sizeof(dolheader));
|
|
if (dol_header == NULL)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
ret = WDVD_Read(dol_header, sizeof(dolheader), (doloffset<<2));
|
|
|
|
entrypoint = load_dol_start(dol_header);
|
|
|
|
if (entrypoint == 0)
|
|
{
|
|
free(dol_header);
|
|
return -1;
|
|
}
|
|
|
|
void *offset;
|
|
u32 pos;
|
|
u32 len;
|
|
|
|
while (load_dol_image_modified(&offset, &pos, &len))
|
|
{
|
|
if (len != 0)
|
|
{
|
|
ret = WDVD_Read(offset, len, (doloffset<<2) + pos);
|
|
|
|
DCFlushRange(offset, len);
|
|
|
|
gamepatches(offset, len, videoSelected, patchcountrystring, vipatch);
|
|
|
|
DCFlushRange(offset, len);
|
|
|
|
Remove_001_Protection(offset, len);
|
|
}
|
|
}
|
|
|
|
free(dol_header);
|
|
|
|
return entrypoint;
|
|
}
|