2010-12-03 23:48:51 +01:00
|
|
|
#include "meta.h"
|
|
|
|
#include "../util.h"
|
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
|
2019-03-11 11:58:57 +01:00
|
|
|
/* MTAF - found in Metal Gear Solid 3: Snake Eater (PS2), Subsistence and HD too */
|
|
|
|
VGMSTREAM * init_vgmstream_mtaf(STREAMFILE *streamFile) {
|
2010-12-03 23:48:51 +01:00
|
|
|
VGMSTREAM * vgmstream = NULL;
|
2011-05-07 13:05:05 +02:00
|
|
|
off_t start_offset;
|
2017-05-18 19:06:22 +02:00
|
|
|
int loop_flag, channel_count;
|
|
|
|
int32_t loop_start, loop_end;
|
2011-05-07 13:05:05 +02:00
|
|
|
|
|
|
|
|
2019-03-11 11:58:57 +01:00
|
|
|
/* checks */
|
2017-05-18 19:06:22 +02:00
|
|
|
if ( !check_extensions(streamFile,"mtaf"))
|
2011-05-07 13:05:05 +02:00
|
|
|
goto fail;
|
2017-05-18 19:06:22 +02:00
|
|
|
if (read_32bitBE(0x00, streamFile) != 0x4d544146) /* "MTAF" */
|
|
|
|
goto fail;
|
|
|
|
/* 0x04(4): pseudo file size (close but smaller) */
|
|
|
|
/* 0x08(4): version? (0), 0x0c(20): null, 0x30(32): some kind of id or config? */
|
2011-05-07 13:05:05 +02:00
|
|
|
|
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
/* HEAD chunk */
|
|
|
|
if (read_32bitBE(0x40, streamFile) != 0x48454144) /* "HEAD" */
|
2011-05-07 13:05:05 +02:00
|
|
|
goto fail;
|
2017-05-18 19:06:22 +02:00
|
|
|
if (read_32bitLE(0x44, streamFile) != 0xB0) /* HEAD size */
|
2011-05-07 13:05:05 +02:00
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
/* 0x48(4): null, 0x4c: usually channel count (sometimes 0x10 with 2ch), 0x50(4): 0x7F (vol?), 0x54(2): 0x40 (pan?) */
|
|
|
|
channel_count = 2 * read_8bit(0x61, streamFile); /* 0x60(4): full block size (0x110 * channels), but this works */
|
|
|
|
/* 0x70(4): ? (00/05/07), 0x80 .. 0xf8: null */
|
2011-05-07 13:05:05 +02:00
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
loop_start = read_32bitLE(0x58, streamFile);
|
|
|
|
loop_end = read_32bitLE(0x5c, streamFile);
|
|
|
|
loop_flag = (loop_start != loop_end);
|
|
|
|
|
|
|
|
/* check loop points vs frame counts */
|
|
|
|
if (loop_start/0x100 != read_32bitLE(0x64, streamFile) ||
|
|
|
|
loop_end /0x100 != read_32bitLE(0x68, streamFile) ) {
|
|
|
|
VGM_LOG("MTAF: wrong loop points\n");
|
2011-05-07 13:05:05 +02:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
/* TRKP chunks (x16) */
|
|
|
|
/* just seem to contain pan/vol stuff (0x7f/0x40), one TRKP with data per channel and the rest fixed values */
|
2011-05-07 13:05:05 +02:00
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
/* DATA chunk */
|
|
|
|
if (read_32bitBE(0x7f8, streamFile) != 0x44415441) /* "DATA" */
|
2011-05-07 13:05:05 +02:00
|
|
|
goto fail;
|
2017-05-18 19:06:22 +02:00
|
|
|
/* 0x7fc: data size (without blocks in case of blocked layout) */
|
2011-05-07 13:05:05 +02:00
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
start_offset = 0x800;
|
2011-05-07 13:05:05 +02:00
|
|
|
|
2019-03-11 11:58:57 +01:00
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
/* build the VGMSTREAM */
|
2011-05-07 13:05:05 +02:00
|
|
|
vgmstream = allocate_vgmstream(channel_count, loop_flag);
|
2010-12-03 23:48:51 +01:00
|
|
|
if (!vgmstream) goto fail;
|
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
vgmstream->sample_rate = 48000; /* always */
|
|
|
|
vgmstream->num_samples = loop_end;
|
2011-05-07 13:05:05 +02:00
|
|
|
vgmstream->loop_start_sample = loop_start;
|
|
|
|
vgmstream->loop_end_sample = loop_end;
|
|
|
|
|
2017-05-18 19:06:22 +02:00
|
|
|
vgmstream->coding_type = coding_MTAF;
|
2011-05-08 05:12:45 +02:00
|
|
|
vgmstream->layout_type = layout_interleave;
|
2019-03-11 11:58:57 +01:00
|
|
|
vgmstream->interleave_block_size = 0x110 / 2; /* kinda hacky for MTAF (stereo codec) track layout */
|
|
|
|
vgmstream->meta_type = meta_MTAF;
|
|
|
|
|
|
|
|
if ( !vgmstream_open_stream(vgmstream, streamFile, start_offset) )
|
|
|
|
goto fail;
|
2011-05-07 13:05:05 +02:00
|
|
|
|
2010-12-03 23:48:51 +01:00
|
|
|
return vgmstream;
|
|
|
|
|
|
|
|
fail:
|
2017-05-18 19:06:22 +02:00
|
|
|
close_vgmstream(vgmstream);
|
2010-12-03 23:48:51 +01:00
|
|
|
return NULL;
|
|
|
|
}
|