2010-04-10 14:06:25 +02:00
|
|
|
#include "meta.h"
|
2017-06-25 02:10:13 +02:00
|
|
|
#include "../coding/coding.h"
|
2010-04-10 14:06:25 +02:00
|
|
|
|
2017-06-25 02:10:13 +02:00
|
|
|
/* P3D - from Radical's Prototype 1/2 (PC/PS3/X360) */
|
2010-04-10 14:06:25 +02:00
|
|
|
VGMSTREAM * init_vgmstream_p3d(STREAMFILE *streamFile) {
|
|
|
|
VGMSTREAM * vgmstream = NULL;
|
2017-08-12 11:46:28 +02:00
|
|
|
off_t start_offset, parse_offset, name_offset = 0;
|
2017-06-25 02:10:13 +02:00
|
|
|
size_t header_size, file_size, data_size;
|
|
|
|
int loop_flag = 0, channel_count, sample_rate, codec;
|
|
|
|
int i, name_count, text_len, block_size = 0, block_count = 0, num_samples;
|
|
|
|
int32_t (*read_32bit)(off_t,STREAMFILE*) = NULL;
|
|
|
|
|
2010-04-10 14:06:25 +02:00
|
|
|
|
|
|
|
/* check extension, case insensitive */
|
2017-06-25 02:10:13 +02:00
|
|
|
if (!check_extensions(streamFile,"p3d"))
|
|
|
|
goto fail;
|
2010-04-10 14:06:25 +02:00
|
|
|
|
|
|
|
/* check header */
|
2017-06-25 02:10:13 +02:00
|
|
|
if (read_32bitBE(0x0,streamFile) != 0x503344FF && /* "P3D"\FF (LE: PC) */
|
|
|
|
read_32bitBE(0x0,streamFile) != 0xFF443350) /* \FF"D3P" (BE: PS3, X360) */
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
read_32bit = read_32bitBE(0x0,streamFile) == 0xFF443350 ? read_32bitBE : read_32bitLE;
|
2010-04-10 14:06:25 +02:00
|
|
|
file_size = get_streamfile_size(streamFile);
|
|
|
|
|
2017-06-25 02:10:13 +02:00
|
|
|
/* base header */
|
|
|
|
header_size = read_32bit(0x4,streamFile);
|
|
|
|
if (0x0C != header_size) goto fail;
|
|
|
|
if (read_32bit(0x08,streamFile) != file_size) goto fail;
|
|
|
|
if (read_32bit(0x0C,streamFile) != 0xFE000000) goto fail; /* fixed */
|
|
|
|
if (read_32bit(0x10,streamFile) + header_size != file_size) goto fail;
|
|
|
|
if (read_32bit(0x14,streamFile) + header_size != file_size) goto fail; /* body size again */
|
|
|
|
if (read_32bit(0x18,streamFile) != 0x0000000A) goto fail; /* fixed */
|
2010-04-10 14:06:25 +02:00
|
|
|
|
2017-06-25 02:10:13 +02:00
|
|
|
/* header text */
|
2010-04-10 14:06:25 +02:00
|
|
|
parse_offset = 0x1C;
|
2017-06-25 02:10:13 +02:00
|
|
|
text_len = read_32bit(parse_offset,streamFile);
|
|
|
|
if (9 != text_len) goto fail;
|
|
|
|
parse_offset += 4;
|
|
|
|
|
2017-06-25 02:21:25 +02:00
|
|
|
/* check the type as P3D is just a generic container used in Radical's games */
|
2017-06-25 02:10:13 +02:00
|
|
|
if (read_32bitBE(parse_offset+0x00,streamFile) != 0x41756469 ||
|
|
|
|
read_32bitBE(parse_offset+0x04,streamFile) != 0x6F46696C ||
|
|
|
|
read_16bitBE(parse_offset+0x08,streamFile) != 0x6500) goto fail; /* "AudioFile\0" */
|
|
|
|
parse_offset += text_len + 1;
|
|
|
|
|
|
|
|
/* file names: always 2 (repeated); but if it's 3 there is an extra string later */
|
|
|
|
name_count = read_32bit(parse_offset,streamFile);
|
|
|
|
if (name_count != 2 && name_count != 3) goto fail; /* 2: Prototype1, 3: Prototype2 */
|
|
|
|
parse_offset += 4;
|
|
|
|
|
2017-08-12 11:46:28 +02:00
|
|
|
/* skip names */
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
if (!name_offset)
|
|
|
|
name_offset = parse_offset + 4;
|
|
|
|
text_len = read_32bit(parse_offset,streamFile) + 1; /* null-terminated */
|
|
|
|
parse_offset += 4 + text_len;
|
2010-04-10 14:06:25 +02:00
|
|
|
}
|
2017-06-25 02:10:13 +02:00
|
|
|
|
2010-04-10 14:06:25 +02:00
|
|
|
/* info count? */
|
2017-06-25 02:10:13 +02:00
|
|
|
if (0x01 != read_32bit(parse_offset,streamFile)) goto fail;
|
2010-04-10 14:06:25 +02:00
|
|
|
parse_offset += 4;
|
2017-06-25 02:10:13 +02:00
|
|
|
|
|
|
|
/* next string can be used as a codec id */
|
|
|
|
text_len = read_32bit(parse_offset,streamFile);
|
|
|
|
codec = read_32bitBE(parse_offset+4,streamFile);
|
|
|
|
parse_offset += 4 + text_len + 1;
|
|
|
|
|
|
|
|
/* extra "Music" string in Prototype 2 */
|
|
|
|
if (name_count == 3) {
|
2017-08-12 11:46:28 +02:00
|
|
|
text_len = read_32bit(parse_offset,streamFile) + 1; /* null-terminated */
|
|
|
|
parse_offset += 4 + text_len;
|
2010-04-10 14:06:25 +02:00
|
|
|
}
|
|
|
|
|
2017-06-25 02:10:13 +02:00
|
|
|
|
|
|
|
/* sub-header per codec */
|
|
|
|
switch(codec) {
|
|
|
|
case 0x72616470: /* "radp" (PC) */
|
|
|
|
if (read_32bitBE(parse_offset,streamFile) != 0x52414450) goto fail; /* "RADP" */
|
|
|
|
parse_offset += 0x04;
|
|
|
|
channel_count = read_32bit(parse_offset+0x00,streamFile);
|
|
|
|
sample_rate = read_32bit(parse_offset+0x04,streamFile);
|
|
|
|
/* 0x08: ? (0x0F) */
|
|
|
|
data_size = read_32bit(parse_offset+0x0c,streamFile);
|
|
|
|
block_size = 0x14;
|
|
|
|
num_samples = data_size / block_size / channel_count * 32;
|
|
|
|
start_offset = parse_offset+0x10;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x6D703300: /* "mp3\0" (PS3) */
|
|
|
|
if ((read_32bitBE(parse_offset,streamFile) & 0xFFFFFF00) != 0x6D703300) goto fail; /* "mp3" */
|
|
|
|
parse_offset += 0x03;
|
|
|
|
/* all fields LE even though the prev parts were BE */
|
|
|
|
sample_rate = read_32bitLE(parse_offset+0x00,streamFile);
|
|
|
|
/* 0x04: mp3 sample rate (ex. @0x00 is 47999 and @0x04 is 48000) */
|
|
|
|
num_samples = read_32bitLE(parse_offset+0x08,streamFile);
|
|
|
|
data_size = read_32bitLE(parse_offset+0x0c,streamFile);
|
|
|
|
channel_count = read_32bitLE(parse_offset+0x10,streamFile);
|
|
|
|
block_size = read_32bitLE(parse_offset+0x14,streamFile);
|
|
|
|
num_samples = num_samples / channel_count; /* total samples */
|
|
|
|
start_offset = parse_offset+0x18;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x786D6100: /* "xma\0" (X360) */
|
|
|
|
if (read_32bitBE(parse_offset,streamFile) != 0x584D4132) goto fail; /* "XMA2" */
|
|
|
|
parse_offset += 0x04;
|
|
|
|
/* 0x00: subheader size? (0x2c), 0x04: seek table size */
|
|
|
|
data_size = read_32bitBE(parse_offset+0x08,streamFile);
|
|
|
|
/* 0x0c: ?, 0x10: ?, 0x14/18: 0x0 */
|
|
|
|
sample_rate = read_32bitBE(parse_offset+0x1c,streamFile);
|
|
|
|
/* 0x20: XMA decoder params, 0x24: abr */
|
|
|
|
block_size = read_32bitBE(parse_offset+0x28,streamFile);
|
|
|
|
num_samples = read_32bitBE(parse_offset+0x2c,streamFile);
|
|
|
|
/* 0x30: original file's samples */
|
|
|
|
block_count = read_32bitBE(parse_offset+0x34,streamFile);
|
|
|
|
channel_count = read_8bit(parse_offset+0x38,streamFile);
|
|
|
|
/* 0x39: channel related? (stream config? channel layout?) */
|
|
|
|
start_offset = parse_offset + 0x3c + read_32bitBE(parse_offset+0x04,streamFile);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-04-21 21:56:49 +02:00
|
|
|
VGM_LOG("P3D: unknown codec 0x%04x\n", codec);
|
2017-06-25 02:10:13 +02:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (start_offset + data_size != file_size) goto fail;
|
|
|
|
|
|
|
|
/* build the VGMSTREAM */
|
2010-04-10 14:06:25 +02:00
|
|
|
vgmstream = allocate_vgmstream(channel_count,loop_flag);
|
|
|
|
if (!vgmstream) goto fail;
|
|
|
|
|
|
|
|
vgmstream->sample_rate = sample_rate;
|
2017-06-25 02:10:13 +02:00
|
|
|
vgmstream->num_samples = num_samples;
|
2010-04-10 14:06:25 +02:00
|
|
|
vgmstream->meta_type = meta_P3D;
|
2017-08-12 11:46:28 +02:00
|
|
|
if (name_offset)
|
|
|
|
read_string(vgmstream->stream_name,STREAM_NAME_SIZE, name_offset,streamFile);
|
2010-04-10 14:06:25 +02:00
|
|
|
|
2017-06-25 02:10:13 +02:00
|
|
|
/* codec init */
|
|
|
|
switch(codec) {
|
|
|
|
case 0x72616470: /* "radp" (PC) */
|
|
|
|
vgmstream->coding_type = coding_RAD_IMA_mono;
|
|
|
|
vgmstream->layout_type = layout_interleave;
|
|
|
|
vgmstream->interleave_block_size = block_size;
|
|
|
|
break;
|
|
|
|
|
|
|
|
#ifdef VGM_USE_MPEG
|
|
|
|
case 0x6D703300: { /* "mp3\0" (PS3) */
|
2017-12-17 19:25:10 +01:00
|
|
|
mpeg_custom_config cfg = {0};
|
2017-06-25 02:10:13 +02:00
|
|
|
|
2017-07-29 23:53:45 +02:00
|
|
|
cfg.interleave = 0x400;
|
2017-12-17 19:25:10 +01:00
|
|
|
cfg.data_size = data_size;
|
2017-07-29 23:53:45 +02:00
|
|
|
/* block_size * 3 = frame size (0x60*3=0x120 or 0x40*3=0xC0) but doesn't seem to have any significance) */
|
2017-06-25 02:10:13 +02:00
|
|
|
|
2018-01-04 23:22:03 +01:00
|
|
|
vgmstream->codec_data = init_mpeg_custom(streamFile, start_offset, &vgmstream->coding_type, vgmstream->channels, MPEG_P3D, &cfg);
|
2017-07-29 23:14:04 +02:00
|
|
|
if (!vgmstream->codec_data) goto fail;
|
2017-12-17 19:25:10 +01:00
|
|
|
vgmstream->layout_type = layout_none;
|
2017-07-29 23:14:04 +02:00
|
|
|
break;
|
2010-04-10 14:06:25 +02:00
|
|
|
}
|
2017-06-25 02:10:13 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef VGM_USE_FFMPEG
|
|
|
|
case 0x786D6100: { /* "xma\0" (X360) */
|
|
|
|
uint8_t buf[0x100];
|
|
|
|
size_t bytes;
|
|
|
|
|
|
|
|
bytes = ffmpeg_make_riff_xma2(buf,0x100, vgmstream->num_samples, data_size, vgmstream->channels, vgmstream->sample_rate, block_count, block_size);
|
|
|
|
if (bytes <= 0) goto fail;
|
|
|
|
|
|
|
|
vgmstream->codec_data = init_ffmpeg_header_offset(streamFile, buf,bytes, start_offset,data_size);
|
|
|
|
if ( !vgmstream->codec_data ) goto fail;
|
|
|
|
vgmstream->coding_type = coding_FFmpeg;
|
|
|
|
vgmstream->layout_type = layout_none;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
2018-04-21 21:56:49 +02:00
|
|
|
VGM_LOG("P3D: unknown codec 0x%04x\n", codec);
|
2017-06-25 02:10:13 +02:00
|
|
|
goto fail;
|
2010-04-10 14:06:25 +02:00
|
|
|
}
|
2017-06-25 02:10:13 +02:00
|
|
|
|
|
|
|
if (!vgmstream_open_stream(vgmstream,streamFile,start_offset))
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
return vgmstream;
|
2010-04-10 14:06:25 +02:00
|
|
|
|
|
|
|
fail:
|
2017-06-25 02:10:13 +02:00
|
|
|
close_vgmstream(vgmstream);
|
2010-04-10 14:06:25 +02:00
|
|
|
return NULL;
|
|
|
|
}
|