mirror of
https://github.com/vgmstream/vgmstream.git
synced 2024-12-19 18:05:52 +01:00
180 lines
5.9 KiB
C
180 lines
5.9 KiB
C
#include "meta.h"
|
|
#include "../coding/coding.h"
|
|
#include "../util/endianness.h"
|
|
#include "../util/chunks.h"
|
|
|
|
|
|
typedef enum { MSADPCM, DSP, MP3, XMA2 } ckd_codec;
|
|
|
|
/* CKD RIFF - UbiArt Framework (v1) audio [Rayman Origins (Wii/X360/PS3/PC)] */
|
|
VGMSTREAM* init_vgmstream_ubi_ckd(STREAMFILE* sf) {
|
|
VGMSTREAM* vgmstream = NULL;
|
|
off_t start_offset, first_offset = 0x0c, chunk_offset;
|
|
size_t chunk_size, data_size;
|
|
int loop_flag, channels, interleave = 0, format;
|
|
ckd_codec codec;
|
|
int big_endian;
|
|
uint32_t (*read_u32)(off_t,STREAMFILE*);
|
|
uint16_t (*read_u16)(off_t,STREAMFILE*);
|
|
|
|
/* checks */
|
|
if (!is_id32be(0x00,sf, "RIFF"))
|
|
goto fail;
|
|
if (!is_id32be(0x08,sf, "WAVE"))
|
|
goto fail;
|
|
|
|
/* .wav.ckd: main (other files are called .xxx.ckd too) */
|
|
if (!check_extensions(sf,"ckd"))
|
|
goto fail;
|
|
|
|
/* another slighly funny RIFF, mostly standard except machine endian and minor oddities */
|
|
if (!is_id32be(0x0c,sf, "fmt "))
|
|
goto fail;
|
|
|
|
big_endian = guess_endian32(0x04, sf);
|
|
read_u32 = big_endian ? read_u32be : read_u32le;
|
|
read_u16 = big_endian ? read_u16be : read_u16le;
|
|
|
|
if (read_u32(0x04, sf) + 0x04 + 0x04 != get_streamfile_size(sf))
|
|
goto fail;
|
|
|
|
loop_flag = 0;
|
|
format = read_u16(0x14,sf);
|
|
channels = read_u16(0x16,sf);
|
|
|
|
switch(format) {
|
|
case 0x0002:
|
|
if (big_endian) {
|
|
if (read_u32(0x26,sf) != 0x6473704C) /* "dspL" */
|
|
goto fail;
|
|
|
|
/* find data chunk, in 2 variants */
|
|
if (find_chunk_be(sf, 0x64617453,first_offset,0, &chunk_offset,&chunk_size)) { /* "datS" */
|
|
/* normal interleave */
|
|
start_offset = chunk_offset;
|
|
data_size = chunk_size;
|
|
interleave = 0x08;
|
|
} else if (find_chunk_be(sf, 0x6461744C,first_offset,0, &chunk_offset,&chunk_size)) { /* "datL" */
|
|
/* mono "datL" or full interleave with a "datR" after the "datL" (no check, pretend it exists) */
|
|
start_offset = chunk_offset;
|
|
|
|
/* chunks follow RIFF's spec of "odd sizes treated as even" [Rayman Origins (Wii)-420_hud~sfx_endofmap_discoloop.wav.ckd] */
|
|
if (chunk_size % 0x02 != 0)
|
|
chunk_size += 0x01;
|
|
data_size = chunk_size * channels;
|
|
interleave = (0x4+0x4) + chunk_size; /* don't forget to skip the "datR"+size chunk */
|
|
} else {
|
|
goto fail;
|
|
}
|
|
|
|
codec = DSP;
|
|
}
|
|
else {
|
|
/* PC has MS-ADPCM, same as wav's except without "fact" (recommended by MS), kinda useless
|
|
* but might as well have it here */
|
|
if (find_chunk_le(sf, 0x64617461,first_offset,0, &chunk_offset,&chunk_size)) { /* "data" */
|
|
start_offset = chunk_offset;
|
|
data_size = chunk_size;
|
|
} else {
|
|
goto fail;
|
|
}
|
|
|
|
interleave = read_u16(0x20, sf);
|
|
if (!msadpcm_check_coefs(sf, 0x28))
|
|
goto fail;
|
|
|
|
/* there is also a "smpl" chunk with full loops too, but other codecs don't have it for the same tracks... */
|
|
|
|
codec = MSADPCM;
|
|
}
|
|
break;
|
|
|
|
case 0x0055:
|
|
if (read_u32(0x26,sf) != 0x6D736620) /* "msf " */
|
|
goto fail;
|
|
start_offset = 0x26;
|
|
data_size = read_u32(0x2A,sf);
|
|
codec = MP3;
|
|
break;
|
|
|
|
case 0x0166:
|
|
if (read_u32(0x48,sf) != 0x7365656B && /* "seek */
|
|
read_u32(0x48,sf) != 0x7365656B) /* "data" */
|
|
goto fail;
|
|
|
|
if (find_chunk_be(sf, 0x64617461,first_offset,0, &chunk_offset,&chunk_size)) { /* "data" */
|
|
start_offset = chunk_offset;
|
|
data_size = chunk_size;
|
|
} else {
|
|
goto fail;
|
|
}
|
|
|
|
codec = XMA2;
|
|
break;
|
|
|
|
default:
|
|
goto fail;
|
|
}
|
|
|
|
|
|
/* build the VGMSTREAM */
|
|
vgmstream = allocate_vgmstream(channels,loop_flag);
|
|
if (!vgmstream) goto fail;
|
|
|
|
vgmstream->sample_rate = read_u32(0x18,sf);
|
|
vgmstream->num_samples = dsp_bytes_to_samples(data_size, channels);
|
|
|
|
vgmstream->coding_type = coding_NGC_DSP;
|
|
vgmstream->meta_type = meta_UBI_CKD;
|
|
|
|
switch(codec) {
|
|
case MSADPCM:
|
|
vgmstream->coding_type = coding_MSADPCM;
|
|
vgmstream->layout_type = layout_none;
|
|
vgmstream->frame_size = interleave;
|
|
break;
|
|
|
|
case DSP:
|
|
vgmstream->layout_type = layout_interleave;
|
|
vgmstream->interleave_block_size = interleave;
|
|
dsp_read_coefs_be(vgmstream,sf, 0x4A, (4+4)+0x60);
|
|
break;
|
|
|
|
#ifdef VGM_USE_MPEG
|
|
case MP3: {
|
|
vgmstream->codec_data = init_mpeg(sf, start_offset, &vgmstream->coding_type, vgmstream->channels);
|
|
if (!vgmstream->codec_data) goto fail;
|
|
vgmstream->layout_type = layout_none;
|
|
|
|
vgmstream->num_samples = mpeg_bytes_to_samples(data_size, vgmstream->codec_data);
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
#ifdef VGM_USE_FFMPEG
|
|
case XMA2: {
|
|
vgmstream->codec_data = init_ffmpeg_xma_chunk(sf, start_offset, data_size, 0x14, 0x34);
|
|
if (!vgmstream->codec_data) goto fail;
|
|
vgmstream->coding_type = coding_FFmpeg;
|
|
vgmstream->layout_type = layout_none;
|
|
|
|
vgmstream->num_samples = read_u32(0x14+0x18,sf);
|
|
|
|
xma_fix_raw_samples(vgmstream, sf, start_offset,data_size, 0, 0,0); /* should apply to num_samples? */
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
default:
|
|
goto fail;
|
|
}
|
|
|
|
if (!vgmstream_open_stream(vgmstream,sf,start_offset))
|
|
goto fail;
|
|
return vgmstream;
|
|
|
|
fail:
|
|
close_vgmstream(vgmstream);
|
|
return NULL;
|
|
}
|