212 lines
4.6 KiB
C
Raw Normal View History

2012-01-21 20:57:41 +00:00
/***************************************************************************
* Copyright (C) 2011
* by Dimok
* Modifications by xFede
* Wiiflowized and heavily improved by Miigotu
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any
* damages arising from the use of this software.
*
* Permission is granted to anyone to use this software for any
* purpose, including commercial applications, and to alter it and
* redistribute it freely, subject to the following restrictions:
*
* 1. The origin of this software must not be misrepresented; you
* must not claim that you wrote the original software. If you use
* this software in a product, an acknowledgment in the product
* documentation would be appreciated but is not required.
*
* 2. Altered source versions must be plainly marked as such, and
* must not be misrepresented as being the original software.
*
* 3. This notice may not be removed or altered from any source
* distribution.
***************************************************************************/
#include <gctypes.h>
#include <malloc.h>
#include <string.h>
#include <stdio.h>
2012-01-21 20:57:41 +00:00
#include "cios.h"
2012-01-21 20:57:41 +00:00
#include "utils.h"
#include "gecko.h"
2012-01-21 20:57:41 +00:00
#include "fs.h"
#include "mload.h"
2012-01-21 20:57:41 +00:00
static bool checked = false;
static bool neek = false;
extern u32 get_ios_base();
bool neek2o(void)
{
if(!checked)
{
u32 num = 0;
neek = !(ISFS_ReadDir("/sneek", NULL, &num));
gprintf("WiiFlow is in %s mode\n", neek ? "neek2o" : "real nand");
checked = true;
}
return neek;
}
2012-01-21 20:57:41 +00:00
/* Check if the cIOS is a D2X. */
bool D2X(u8 ios, u8 *base)
2012-01-21 20:57:41 +00:00
{
if(neek2o())
{
*base = (u8)IOS_GetVersion();
return true;
}
2012-01-21 20:57:41 +00:00
iosinfo_t *info = GetInfo(ios);
if(!info)
return false;
if(info->magicword != 0x1ee7c105 /* Magic Word */
|| info->magicversion != 1 /* Magic Version */
|| info->version < 6 /* Version */
|| strncasecmp(info->name, "d2x", 3) != 0) /* Name */
{
free(info);
return false;
}
*base = (u8)info->baseios;
free(info);
return true;
2012-01-21 20:57:41 +00:00
}
signed_blob *GetTMD(u8 ios, u32 *TMD_Length)
{
if(ES_GetStoredTMDSize(TITLE_ID(1, ios), TMD_Length) < 0)
return NULL;
signed_blob *TMD = (signed_blob*)memalign(32, ALIGN32(*TMD_Length));
if(TMD == NULL)
return NULL;
if(ES_GetStoredTMD(TITLE_ID(1, ios), TMD, *TMD_Length) < 0)
{
free(TMD);
return NULL;
}
return TMD;
}
2012-01-21 20:57:41 +00:00
/*
* Reads the ios info struct from the .app file.
* @return pointer to iosinfo_t on success else NULL. The user is responsible for freeing the buffer.
*/
iosinfo_t *GetInfo(u8 ios)
2012-01-21 20:57:41 +00:00
{
u32 TMD_Length;
signed_blob *TMD = GetTMD(ios, &TMD_Length);
if(TMD == NULL)
2012-01-21 20:57:41 +00:00
return NULL;
char filepath[ISFS_MAXPATH] ATTRIBUTE_ALIGN(32);
sprintf(filepath, "/title/00000001/%08x/content/%08x.app", ios, *(u8 *)((u32)TMD+0x1E7));
2012-01-21 20:57:41 +00:00
free(TMD);
2012-01-21 20:57:41 +00:00
u32 size = 0;
2012-05-13 17:25:26 +00:00
u8 *buffer = ISFS_GetFile((u8 *)filepath, &size, sizeof(iosinfo_t));
if(buffer == NULL || size == 0)
return NULL;
2012-01-21 20:57:41 +00:00
2012-05-13 17:25:26 +00:00
iosinfo_t *iosinfo = (iosinfo_t *)buffer;
2012-01-21 20:57:41 +00:00
return iosinfo;
}
u32 Title_GetSize_FromTMD(tmd *tmd_data)
{
u32 cnt, size = 0;
/* Calculate title size */
for (cnt = 0; cnt < tmd_data->num_contents; cnt++) {
tmd_content *content = &tmd_data->contents[cnt];
/* Add content size */
size += content->size;
}
return size;
}
int get_ios_type(u8 slot)
{
u32 TMD_Length;
signed_blob *TMD_Buffer = GetTMD(slot, &TMD_Length);
if(TMD_Buffer == NULL)
return IOS_TYPE_NO_CIOS;
tmd *iosTMD = (tmd*)SIGNATURE_PAYLOAD(TMD_Buffer);
if(Title_GetSize_FromTMD(iosTMD) < 0x100000 || iosTMD->title_version == 65280)
{
free(TMD_Buffer);
return IOS_TYPE_NO_CIOS;
}
iosinfo_t *info = GetInfo(slot);
if(info == NULL)
{
free(TMD_Buffer);
return IOS_TYPE_NO_CIOS;
}
free(info);
u8 base = 0;
switch(slot)
{
case 222:
case 223:
case 224:
case 225:
if(iosTMD->title_version == 1)
return IOS_TYPE_KWIIRK;
else
return IOS_TYPE_HERMES;
case 245:
case 246:
case 247:
case 248:
case 249:
case 250:
case 251:
if(D2X(slot, &base))
return IOS_TYPE_D2X;
else
return IOS_TYPE_WANIN;
default:
if(D2X(slot, &base))
return IOS_TYPE_D2X;
else
return IOS_TYPE_NO_CIOS;
}
free(TMD_Buffer);
}
int is_ios_type(int type, u8 slot)
{
return (get_ios_type(slot) == type);
}
bool shadow_mload()
{
if(!is_ios_type(IOS_TYPE_HERMES, IOS_GetVersion()))
return false;
int v51 = (5 << 4) & 1;
if (mload_get_version() >= v51)
{
// shadow /dev/mload supported in hermes cios v5.1
//IOS_Open("/dev/usb123/OFF",0);// this disables ehc completely
IOS_Open("/dev/mload/OFF",0);
gprintf("Shadow mload\n");
return true;
}
return false;
}