2010-12-27 09:44:27 +00:00
|
|
|
#include <malloc.h>
|
|
|
|
#include <stdio.h>
|
2011-11-13 09:09:27 +00:00
|
|
|
#include "Channels/channels.h"
|
2010-12-27 09:44:27 +00:00
|
|
|
#include "usbloader/disc.h"
|
|
|
|
#include "usbloader/wbfs.h"
|
|
|
|
#include "utils/uncompress.h"
|
|
|
|
#include "OpeningBNR.hpp"
|
|
|
|
|
|
|
|
BNRInstance * BNRInstance::instance = NULL;
|
|
|
|
|
|
|
|
OpeningBNR::OpeningBNR()
|
2011-07-25 22:28:22 +00:00
|
|
|
: imetHdr(0)
|
2010-12-27 09:44:27 +00:00
|
|
|
{
|
|
|
|
memset(gameID, 0, sizeof(gameID));
|
|
|
|
}
|
|
|
|
|
|
|
|
OpeningBNR::~OpeningBNR()
|
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
if(imetHdr)
|
|
|
|
free(imetHdr);
|
2010-12-27 09:44:27 +00:00
|
|
|
}
|
|
|
|
|
2011-12-22 22:44:48 +00:00
|
|
|
bool OpeningBNR::Load(const u64 &tid, const char *pathPrefix)
|
2011-11-13 09:09:27 +00:00
|
|
|
{
|
|
|
|
if(tid == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
u32 tidLow = (u32) (tid & 0xFFFFFFFF);
|
|
|
|
char id[6];
|
|
|
|
memset(id, 0, sizeof(id));
|
|
|
|
memcpy(id, &tidLow, 4);
|
|
|
|
|
|
|
|
if(memcmp(gameID, id, 6) == 0)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
memcpy(gameID, id, 6);
|
|
|
|
|
|
|
|
if(imetHdr)
|
|
|
|
free(imetHdr);
|
|
|
|
imetHdr = NULL;
|
|
|
|
|
2011-12-22 22:44:48 +00:00
|
|
|
imetHdr = (IMETHeader*) Channels::GetOpeningBnr(tid, pathPrefix);
|
2011-11-13 09:09:27 +00:00
|
|
|
if(!imetHdr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (imetHdr->fcc != 'IMET')
|
|
|
|
{
|
|
|
|
free(imetHdr);
|
|
|
|
imetHdr = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-12-27 09:44:27 +00:00
|
|
|
bool OpeningBNR::Load(const u8 * discid)
|
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
if(!discid)
|
|
|
|
return false;
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
if(memcmp(gameID, discid, 6) == 0)
|
|
|
|
return true;
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-11-13 09:09:27 +00:00
|
|
|
memcpy(gameID, discid, 6);
|
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
if(imetHdr)
|
|
|
|
free(imetHdr);
|
|
|
|
imetHdr = NULL;
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
wbfs_disc_t *disc = WBFS_OpenDisc((u8 *) gameID);
|
|
|
|
if (!disc)
|
|
|
|
return false;
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
wiidisc_t *wdisc = wd_open_disc((int(*)(void *, u32, u32, void *)) wbfs_disc_read, disc);
|
|
|
|
if (!wdisc)
|
|
|
|
{
|
|
|
|
WBFS_CloseDisc(disc);
|
|
|
|
return false;
|
|
|
|
}
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
imetHdr = (IMETHeader*) wd_extract_file(wdisc, ALL_PARTITIONS, (char *) "opening.bnr");
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
wd_close_disc(wdisc);
|
|
|
|
WBFS_CloseDisc(disc);
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
if(!imetHdr)
|
|
|
|
return false;
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
if (imetHdr->fcc != 'IMET')
|
|
|
|
{
|
|
|
|
free(imetHdr);
|
|
|
|
imetHdr = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
return true;
|
2010-12-27 09:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const u16 * OpeningBNR::GetIMETTitle(int lang)
|
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
if(!imetHdr || lang < 0 || lang >= 10)
|
|
|
|
return NULL;
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
if(imetHdr->names[lang][0] == 0)
|
|
|
|
lang = CONF_LANG_ENGLISH;
|
2010-12-27 09:44:27 +00:00
|
|
|
|
2011-07-25 22:28:22 +00:00
|
|
|
return imetHdr->names[lang];
|
2010-12-27 09:44:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const u8 * OpeningBNR::GetBannerSound(u32 * size)
|
|
|
|
{
|
2011-07-25 22:28:22 +00:00
|
|
|
if(!imetHdr)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
const U8Header *bnrArcHdr = (U8Header *) (imetHdr + 1);
|
|
|
|
const U8Entry *fst = (const U8Entry *) (((const u8 *) bnrArcHdr) + bnrArcHdr->rootNodeOffset);
|
|
|
|
|
|
|
|
u32 i;
|
|
|
|
for (i = 1; i < fst[0].numEntries; ++i)
|
|
|
|
if (fst[i].fileType == 0 && strcasecmp(u8Filename(fst, i), "sound.bin") == 0) break;
|
|
|
|
|
|
|
|
if (i >= fst[0].numEntries)
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
const u8 *sound_bin = ((const u8 *) bnrArcHdr) + fst[i].fileOffset;
|
|
|
|
if (((IMD5Header *) sound_bin)->fcc != 'IMD5')
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
const u8 *soundChunk = sound_bin + sizeof(IMD5Header);
|
|
|
|
u32 soundChunkSize = fst[i].fileLength - sizeof(IMD5Header);
|
|
|
|
|
|
|
|
if (*((u32*) soundChunk) == 'LZ77')
|
|
|
|
{
|
|
|
|
u32 uncSize = 0;
|
|
|
|
u8 * uncompressed_data = uncompressLZ77(soundChunk, soundChunkSize, &uncSize);
|
|
|
|
if (!uncompressed_data)
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (size) *size = uncSize;
|
|
|
|
|
|
|
|
return uncompressed_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 *out = (u8 *) malloc(soundChunkSize);
|
|
|
|
if (out)
|
|
|
|
{
|
|
|
|
memcpy(out, soundChunk, soundChunkSize);
|
|
|
|
if (size) *size = soundChunkSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
2010-12-27 09:44:27 +00:00
|
|
|
}
|