mirror of
https://github.com/vgmstream/vgmstream.git
synced 2025-01-11 12:32:08 +01:00
129 lines
4.3 KiB
C
129 lines
4.3 KiB
C
#include "meta.h"
|
|
#include "../coding/coding.h"
|
|
|
|
/* .SBK - from Addiction Pinball (PC) */
|
|
VGMSTREAM *init_vgmstream_sbk(STREAMFILE *sf) {
|
|
VGMSTREAM *vgmstream = NULL;
|
|
uint32_t sound_offset, sound_size, padding_size, sample_rate;
|
|
uint16_t format, channels, block_size, bps;
|
|
off_t table_offset, data_offset, entry_offset, cfg_fmt_offset;
|
|
size_t table_size, data_size, cfg_entry_size;
|
|
int target_subsong = sf->stream_index, total_subsongs, loop_flag, is_streamed;
|
|
|
|
/* checks */
|
|
if (!check_extensions(sf, "sbk"))
|
|
goto fail;
|
|
|
|
/* check header */
|
|
if (read_u32be(0x00, sf) != 0x52494646) /* "RIFF" */
|
|
goto fail;
|
|
if (read_u32be(0x08, sf) != 0x53424E4B) /* "SBNK" */
|
|
goto fail;
|
|
|
|
if (!find_chunk_le(sf, 0x57415649, 0x0c, 0, &table_offset, &table_size)) /* "WAVI" */
|
|
goto fail;
|
|
|
|
if (find_chunk_le(sf, 0x53574156, 0x0c, 0, &data_offset, &data_size)) { /* "SWAV" */
|
|
cfg_entry_size = 0x38;
|
|
cfg_fmt_offset = 0x1c;
|
|
} else {
|
|
/* 1997 demo version with sound names and no streamed section */
|
|
cfg_entry_size = 0x24;
|
|
cfg_fmt_offset = 0x0c;
|
|
}
|
|
|
|
total_subsongs = table_size / cfg_entry_size;
|
|
|
|
if (target_subsong == 0) target_subsong = 1;
|
|
if (target_subsong < 0 || target_subsong > total_subsongs || total_subsongs < 1)
|
|
goto fail;
|
|
|
|
entry_offset = table_offset + cfg_entry_size * (target_subsong - 1);
|
|
sound_offset = read_u32le(entry_offset + 0x04, sf);
|
|
sound_size = read_u32le(entry_offset + 0x00, sf);
|
|
if (cfg_entry_size == 0x38) {
|
|
padding_size = read_u32le(entry_offset + 0x10, sf);
|
|
sound_offset += padding_size;
|
|
sound_size -= padding_size;
|
|
|
|
is_streamed = read_u8(entry_offset + 0x36, sf);
|
|
} else {
|
|
is_streamed = 0;
|
|
}
|
|
|
|
if (!is_streamed) {
|
|
if (!find_chunk_le(sf, 0x57415644, 0x0c, 0, &data_offset, &data_size)) /* "WAVD" */
|
|
goto fail;
|
|
} else {
|
|
if (!find_chunk_le(sf, 0x53574156, 0x0c, 0, &data_offset, &data_size)) /* "SWAV" */
|
|
goto fail;
|
|
}
|
|
|
|
sound_offset += data_offset;
|
|
|
|
/* read fmt chunk */
|
|
format = read_u16le(entry_offset + cfg_fmt_offset + 0x00, sf);
|
|
channels = read_u16le(entry_offset + cfg_fmt_offset + 0x02, sf);
|
|
sample_rate = read_u32le(entry_offset + cfg_fmt_offset + 0x04, sf);
|
|
block_size = read_u16le(entry_offset + cfg_fmt_offset + 0x0c, sf);
|
|
bps = read_u16le(entry_offset + cfg_fmt_offset + 0x0e, sf);
|
|
|
|
loop_flag = 0;
|
|
|
|
/* build the VGMSTREAM */
|
|
vgmstream = allocate_vgmstream(channels, loop_flag);
|
|
if (!vgmstream) goto fail;
|
|
|
|
/* fill in the vital statistics */
|
|
vgmstream->meta_type = meta_SBK;
|
|
vgmstream->sample_rate = sample_rate;
|
|
vgmstream->stream_size = sound_size;
|
|
vgmstream->num_streams = total_subsongs;
|
|
|
|
if (cfg_entry_size == 0x24) {
|
|
uint32_t num_entries, i;
|
|
|
|
if (!find_chunk_le(sf, 0x544F4320, 0x0c, 0, &table_offset, &table_size)) /* "TOC " */
|
|
goto fail;
|
|
|
|
num_entries = table_size / 0x10;
|
|
for (i = 0; i < num_entries; i++) {
|
|
entry_offset = table_offset + 0x10 * i;
|
|
|
|
if ((read_u8(entry_offset + 0x01, sf) & 0x80) &&
|
|
read_u8(entry_offset + 0x00, sf) == (target_subsong - 1)) {
|
|
read_string(vgmstream->stream_name, 0x0c, entry_offset + 0x04, sf);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
switch (format) {
|
|
case 0x01: /* PCM */
|
|
if (bps != 8 && bps != 16)
|
|
goto fail;
|
|
|
|
vgmstream->coding_type = (bps == 8) ? coding_PCM8_U : coding_PCM16LE;
|
|
vgmstream->layout_type = layout_interleave;
|
|
vgmstream->interleave_block_size = (bps == 8) ? 0x01 : 0x02;
|
|
vgmstream->num_samples = pcm_bytes_to_samples(sound_size, channels, bps);
|
|
break;
|
|
case 0x11: /* Microsoft IMA */
|
|
vgmstream->coding_type = coding_MS_IMA;
|
|
vgmstream->layout_type = layout_none;
|
|
vgmstream->interleave_block_size = block_size;
|
|
vgmstream->num_samples = ms_ima_bytes_to_samples(sound_size, block_size, channels);
|
|
break;
|
|
default:
|
|
goto fail;
|
|
}
|
|
|
|
if (!vgmstream_open_stream(vgmstream, sf, sound_offset))
|
|
goto fail;
|
|
return vgmstream;
|
|
|
|
fail:
|
|
close_vgmstream(vgmstream);
|
|
return NULL;
|
|
}
|