2008-05-14 20:44:19 +00:00
|
|
|
|
#include "meta.h"
|
|
|
|
|
#include "../util.h"
|
|
|
|
|
|
|
|
|
|
/* MIB
|
|
|
|
|
|
|
|
|
|
PS2 MIB format is a headerless format.
|
|
|
|
|
The interleave value can be found by checking the body of the data.
|
|
|
|
|
|
|
|
|
|
The interleave start allways at offset 0 with a int value (which can have
|
|
|
|
|
many values : 0x0000, 0x0002, 0x0006 etc...) follow by 12 empty (zero) values.
|
|
|
|
|
|
|
|
|
|
The interleave value is the offset where you found the same 16 bytes.
|
|
|
|
|
|
|
|
|
|
The n<EFBFBD> of channels can be found by checking each time you found this 16 bytes.
|
|
|
|
|
|
|
|
|
|
The interleave value can be very "large" (up to 0x20000 found so far) and is allways
|
|
|
|
|
a 0x10 multiply value.
|
|
|
|
|
|
|
|
|
|
The loop values can be found by checking the 'tags' offset (found @ 0x02 each 0x10 bytes).
|
|
|
|
|
06 = start of the loop point (can be found for each channel)
|
|
|
|
|
03 - end of the loop point (can be found for each channel)
|
|
|
|
|
|
|
|
|
|
The .MIH header contains all informations about frequency, numbers of channels, interleave
|
|
|
|
|
but has, afaik, no loop values.
|
|
|
|
|
|
|
|
|
|
known extensions : MIB (MIH for the header) MIC (concatenation of MIB+MIH)
|
|
|
|
|
Nota : the MIC stuff is not supported here as there is
|
|
|
|
|
another MIC format which can be found in Koei Games.
|
|
|
|
|
|
|
|
|
|
2008-05-14 - Fastelbja : First version ...
|
2008-05-20 18:09:05 +00:00
|
|
|
|
2008-05-20 - Fastelbja : Fix loop value when loopEnd==0
|
2008-05-14 20:44:19 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2008-05-20 15:18:38 +00:00
|
|
|
|
VGMSTREAM * init_vgmstream_ps2_mib(STREAMFILE *streamFile) {
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
|
|
|
|
VGMSTREAM * vgmstream = NULL;
|
2008-05-20 15:18:38 +00:00
|
|
|
|
STREAMFILE * streamFileMIH = NULL;
|
|
|
|
|
char filename[260];
|
|
|
|
|
|
2008-05-14 20:44:19 +00:00
|
|
|
|
uint8_t mibBuffer[0x10];
|
|
|
|
|
uint8_t testBuffer[0x10];
|
|
|
|
|
|
|
|
|
|
size_t fileLength;
|
|
|
|
|
|
|
|
|
|
off_t loopStart = 0;
|
|
|
|
|
off_t loopEnd = 0;
|
|
|
|
|
off_t interleave = 0;
|
|
|
|
|
off_t readOffset = 0;
|
|
|
|
|
|
2008-05-20 15:18:38 +00:00
|
|
|
|
char filenameMIH[260];
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
|
|
|
|
uint8_t gotMIH=0;
|
|
|
|
|
|
2008-05-14 21:41:16 +00:00
|
|
|
|
int i, channel_count=1;
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
|
|
|
|
/* check extension, case insensitive */
|
2008-05-20 15:18:38 +00:00
|
|
|
|
streamFile->get_name(streamFile,filename,sizeof(filename));
|
2008-05-14 20:44:19 +00:00
|
|
|
|
if (strcasecmp("mib",filename_extension(filename)) &&
|
|
|
|
|
strcasecmp("mi4",filename_extension(filename))) goto fail;
|
|
|
|
|
|
|
|
|
|
/* check for .MIH file */
|
|
|
|
|
strcpy(filenameMIH,filename);
|
|
|
|
|
strcpy(filenameMIH+strlen(filenameMIH)-3,"MIH");
|
|
|
|
|
|
2008-05-20 15:18:38 +00:00
|
|
|
|
streamFileMIH = streamFile->open(streamFile,filenameMIH,STREAMFILE_DEFAULT_BUFFER_SIZE);
|
|
|
|
|
if (streamFileMIH) gotMIH = 1;
|
2008-05-14 21:41:16 +00:00
|
|
|
|
|
2008-05-20 15:18:38 +00:00
|
|
|
|
/* Search for interleave value & loop points */
|
2008-05-14 20:44:19 +00:00
|
|
|
|
/* Get the first 16 values */
|
2008-05-20 15:18:38 +00:00
|
|
|
|
fileLength = get_streamfile_size(streamFile);
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
2008-06-03 18:41:26 +00:00
|
|
|
|
readOffset+=(off_t)read_streamfile(mibBuffer,0,0x10,streamFile);
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
|
|
|
|
do {
|
2008-06-03 18:41:26 +00:00
|
|
|
|
readOffset+=(off_t)read_streamfile(testBuffer,readOffset,0x10,streamFile);
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
|
|
|
|
if(!memcmp(testBuffer,mibBuffer,0x10)) {
|
|
|
|
|
if(interleave==0) interleave=readOffset-0x10;
|
|
|
|
|
|
|
|
|
|
// be sure to point to an interleave value
|
2008-05-14 21:13:47 +00:00
|
|
|
|
if(((readOffset-0x10)==channel_count*interleave)) {
|
|
|
|
|
channel_count++;
|
2008-05-14 20:44:19 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Loop Start ...
|
|
|
|
|
if(testBuffer[0x01]==0x06) {
|
|
|
|
|
if(loopStart==0) loopStart = readOffset-0x10;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Loop End ...
|
|
|
|
|
if(testBuffer[0x01]==0x03) {
|
|
|
|
|
if(loopEnd==0) loopEnd = readOffset-0x10;
|
|
|
|
|
}
|
|
|
|
|
|
2008-06-03 18:41:26 +00:00
|
|
|
|
} while (streamFile->get_offset(streamFile)<(int32_t)fileLength);
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
2008-05-17 15:39:02 +00:00
|
|
|
|
if(gotMIH)
|
2008-05-20 15:18:38 +00:00
|
|
|
|
channel_count=read_32bitLE(0x08,streamFileMIH);
|
2008-05-17 15:39:02 +00:00
|
|
|
|
|
2008-05-14 20:44:19 +00:00
|
|
|
|
/* build the VGMSTREAM */
|
2008-05-20 18:09:05 +00:00
|
|
|
|
vgmstream = allocate_vgmstream(channel_count,((loopStart!=0) && (loopEnd!=0)));
|
2008-05-14 20:44:19 +00:00
|
|
|
|
if (!vgmstream) goto fail;
|
|
|
|
|
|
|
|
|
|
/* fill in the vital statistics */
|
|
|
|
|
if(gotMIH) {
|
|
|
|
|
// Read stuff from the MIH file
|
2008-05-20 15:18:38 +00:00
|
|
|
|
vgmstream->channels = read_32bitLE(0x08,streamFileMIH);
|
|
|
|
|
vgmstream->sample_rate = read_32bitLE(0x0C,streamFileMIH);
|
|
|
|
|
vgmstream->interleave_block_size = read_32bitLE(0x10,streamFileMIH);
|
|
|
|
|
vgmstream->num_samples=((read_32bitLE(0x10,streamFileMIH)*
|
|
|
|
|
(read_32bitLE(0x14,streamFileMIH)-1)*2)+
|
|
|
|
|
((read_32bitLE(0x04,streamFileMIH)>>8)*2))/16*28/2;
|
2008-05-14 20:44:19 +00:00
|
|
|
|
} else {
|
|
|
|
|
vgmstream->channels = channel_count;
|
|
|
|
|
vgmstream->interleave_block_size = interleave;
|
|
|
|
|
|
|
|
|
|
if(!strcasecmp("mib",filename_extension(filename)))
|
|
|
|
|
vgmstream->sample_rate = 44100;
|
|
|
|
|
|
|
|
|
|
if(!strcasecmp("mi4",filename_extension(filename)))
|
|
|
|
|
vgmstream->sample_rate = 48000;
|
|
|
|
|
|
2008-06-03 18:41:26 +00:00
|
|
|
|
vgmstream->num_samples = (int32_t)(fileLength/16/channel_count*28);
|
2008-05-14 20:44:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(loopStart!=0) {
|
|
|
|
|
vgmstream->loop_start_sample = ((loopStart/(interleave*channel_count))*interleave)/16*14*channel_count;
|
|
|
|
|
vgmstream->loop_start_sample += (loopStart%(interleave*channel_count))/16*14*channel_count;
|
|
|
|
|
vgmstream->loop_end_sample = ((loopEnd/(interleave*channel_count))*interleave)/16*14*channel_count;
|
|
|
|
|
vgmstream->loop_end_sample += (loopEnd%(interleave*channel_count))/16*14*channel_count;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vgmstream->coding_type = coding_PSX;
|
|
|
|
|
vgmstream->layout_type = layout_interleave;
|
|
|
|
|
|
|
|
|
|
vgmstream->meta_type = meta_PS2_MIB;
|
2008-05-20 15:18:38 +00:00
|
|
|
|
|
2008-05-14 20:44:19 +00:00
|
|
|
|
if (gotMIH) {
|
|
|
|
|
vgmstream->meta_type = meta_PS2_MIB_MIH;
|
2008-05-20 15:18:38 +00:00
|
|
|
|
close_streamfile(streamFileMIH); streamFileMIH=NULL;
|
2008-05-14 20:44:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* open the file for reading by each channel */
|
|
|
|
|
{
|
|
|
|
|
for (i=0;i<channel_count;i++) {
|
2008-05-20 15:18:38 +00:00
|
|
|
|
vgmstream->ch[i].streamfile = streamFile->open(streamFile,filename,0x8000);
|
2008-05-14 20:44:19 +00:00
|
|
|
|
|
|
|
|
|
if (!vgmstream->ch[i].streamfile) goto fail;
|
|
|
|
|
|
|
|
|
|
vgmstream->ch[i].channel_start_offset=
|
|
|
|
|
vgmstream->ch[i].offset=0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return vgmstream;
|
|
|
|
|
|
|
|
|
|
/* clean up anything we may have opened */
|
|
|
|
|
fail:
|
2008-05-20 15:18:38 +00:00
|
|
|
|
if (streamFileMIH) close_streamfile(streamFileMIH);
|
2008-05-14 20:44:19 +00:00
|
|
|
|
if (vgmstream) close_vgmstream(vgmstream);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|