mirror of
https://github.com/vgmstream/vgmstream.git
synced 2025-02-20 20:41:08 +01:00
Add XSSB .bin/lbin [Blinx 2 (Xbox)]
This commit is contained in:
parent
410fe894cd
commit
d803349f6b
@ -1248,6 +1248,7 @@ static const meta_info meta_info_list[] = {
|
||||
{meta_UBI_HX, "Ubisoft HXx header"},
|
||||
{meta_BMP_KONAMI, "Konami BMP header"},
|
||||
{meta_ISB, "Creative ISACT header"},
|
||||
{meta_XSSB, "Artoon XSSB header"},
|
||||
|
||||
};
|
||||
|
||||
|
@ -1778,10 +1778,14 @@
|
||||
RelativePath=".\meta\xps.c"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\meta\xss.c"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\meta\xss.c"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\meta\xssb.c"
|
||||
>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\meta\xvag.c"
|
||||
>
|
||||
|
@ -523,6 +523,7 @@
|
||||
<ClCompile Include="meta\xpcm.c" />
|
||||
<ClCompile Include="meta\xps.c" />
|
||||
<ClCompile Include="meta\xss.c" />
|
||||
<ClCompile Include="meta\xssb.c" />
|
||||
<ClCompile Include="meta\xvag.c" />
|
||||
<ClCompile Include="meta\xmd.c" />
|
||||
<ClCompile Include="meta\xwb.c" />
|
||||
|
@ -1087,6 +1087,9 @@
|
||||
<ClCompile Include="meta\xss.c">
|
||||
<Filter>meta\Source Files</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="meta\xssb.c">
|
||||
<Filter>meta\Source Files</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="meta\xvag.c">
|
||||
<Filter>meta\Source Files</Filter>
|
||||
</ClCompile>
|
||||
|
@ -870,4 +870,6 @@ VGMSTREAM * init_vgmstream_bmp_konami(STREAMFILE * streamFile);
|
||||
|
||||
VGMSTREAM * init_vgmstream_isb(STREAMFILE * streamFile);
|
||||
|
||||
VGMSTREAM* init_vgmstream_xssb(STREAMFILE *sf);
|
||||
|
||||
#endif /*_META_H*/
|
||||
|
140
src/meta/xssb.c
Normal file
140
src/meta/xssb.c
Normal file
@ -0,0 +1,140 @@
|
||||
#include "meta.h"
|
||||
#include "../coding/coding.h"
|
||||
|
||||
|
||||
//todo test and rethink usefulness/viability of globally using this
|
||||
/* generic helper with a usable fields to describe static header values */
|
||||
typedef struct {
|
||||
int codec;
|
||||
int type;
|
||||
int channels;
|
||||
int sample_rate;
|
||||
int loop_flag;
|
||||
int loop_start;
|
||||
int loop_end;
|
||||
|
||||
int total_subsongs;
|
||||
int target_subsong;
|
||||
|
||||
size_t file_size;
|
||||
|
||||
off_t info_start;
|
||||
|
||||
off_t header_start;
|
||||
size_t header_entry;
|
||||
off_t header_offset;
|
||||
|
||||
off_t data_start;
|
||||
size_t data_size;
|
||||
|
||||
off_t stream_start;
|
||||
size_t stream_size;
|
||||
} header_t;
|
||||
|
||||
/* XSSB - from Artoon games [Blinx (Xbox), Blinx 2 (Xbox)] */
|
||||
VGMSTREAM* init_vgmstream_xssb(STREAMFILE *sf) {
|
||||
VGMSTREAM *vgmstream = NULL;
|
||||
//off_t start_offset, header_offset, data_start, info_start, header_start;
|
||||
//size_t header_size, stream_size;
|
||||
//int loop_flag, channel_count, sample_rate, codec, loop_start, loop_end;
|
||||
//int total_subsongs, target_subsong = streamFile->stream_index;
|
||||
header_t h;
|
||||
|
||||
|
||||
/* checks */
|
||||
/* .bin: from named files inside .ipk bigfiles */
|
||||
if (!check_extensions(sf, "bin,lbin"))
|
||||
goto fail;
|
||||
|
||||
if (read_u32be(0x00, sf) != 0x58535342) /* "XSSB" */
|
||||
goto fail;
|
||||
/* 0x04: null */
|
||||
/* 0x08: date-version ('20011217' in hex) */
|
||||
/* 0x0c: null */
|
||||
|
||||
h.info_start = read_s32le(0x10, sf);
|
||||
h.header_start = read_s32le(0x14, sf);
|
||||
h.data_start = read_s32le(0x18, sf);
|
||||
/* 0x1c: null */
|
||||
|
||||
h.header_entry = read_s16le(h.info_start + 0x00, sf);
|
||||
/* 0x02: always 127 */
|
||||
|
||||
/* get subsongs from header entries */
|
||||
{
|
||||
off_t offset = h.header_start;
|
||||
|
||||
h.total_subsongs = 0;
|
||||
h.target_subsong = sf->stream_index <= 0 ? 1 : sf->stream_index;
|
||||
|
||||
h.header_offset = 0;
|
||||
while (offset < h.data_start) {
|
||||
/* headers are just pasted together and then padding */
|
||||
if (read_u32be(offset, sf) == 0)
|
||||
break;
|
||||
h.total_subsongs++;
|
||||
|
||||
if (h.target_subsong == h.total_subsongs) {
|
||||
h.header_offset = offset;
|
||||
}
|
||||
|
||||
offset += h.header_entry;
|
||||
}
|
||||
|
||||
if (h.header_offset == 0)
|
||||
goto fail;
|
||||
if (h.target_subsong > h.total_subsongs || h.total_subsongs < 1)
|
||||
goto fail;
|
||||
}
|
||||
|
||||
/* read header */
|
||||
h.codec = read_s16le(h.header_offset + 0x00, sf);
|
||||
h.channels = read_s16le(h.header_offset + 0x02, sf);
|
||||
h.sample_rate = read_u16le(h.header_offset + 0x04, sf);
|
||||
/* 0x08: bitrate */
|
||||
/* 0x0c: block align/bps */
|
||||
/* 0x10: 0=PCM, 2=XBOX-IMA? */
|
||||
h.stream_start = read_s32le(h.header_offset + 0x14, sf) + h.data_start;
|
||||
h.stream_size = read_s32le(h.header_offset + 0x18, sf);
|
||||
h.loop_start = read_s32le(h.header_offset + 0x1c, sf);
|
||||
h.loop_end = read_s32le(h.header_offset + 0x20, sf);
|
||||
/* others: unknown and mostly fixed values */
|
||||
h.loop_flag = (h.loop_end > 0);
|
||||
|
||||
|
||||
/* build the VGMSTREAM */
|
||||
vgmstream = allocate_vgmstream(h.channels, h.loop_flag);
|
||||
if (!vgmstream) goto fail;
|
||||
|
||||
vgmstream->meta_type = meta_XSSB;
|
||||
vgmstream->sample_rate = h.sample_rate;
|
||||
vgmstream->loop_start_sample = h.loop_start;
|
||||
vgmstream->loop_end_sample = h.loop_end;
|
||||
vgmstream->num_streams = h.total_subsongs;
|
||||
vgmstream->stream_size = h.stream_size;
|
||||
|
||||
switch(h.codec) {
|
||||
case 0x01:
|
||||
vgmstream->coding_type = coding_PCM16LE;
|
||||
vgmstream->layout_type = layout_interleave;
|
||||
vgmstream->interleave_block_size = 0x01;
|
||||
|
||||
vgmstream->num_samples = pcm_bytes_to_samples(h.stream_size, h.channels, 16);
|
||||
break;
|
||||
|
||||
case 0x69:
|
||||
vgmstream->coding_type = coding_XBOX_IMA;
|
||||
vgmstream->layout_type = layout_none;
|
||||
|
||||
vgmstream->num_samples = xbox_ima_bytes_to_samples(h.stream_size, h.channels);
|
||||
break;
|
||||
}
|
||||
|
||||
if (!vgmstream_open_stream(vgmstream, sf, h.stream_start))
|
||||
goto fail;
|
||||
return vgmstream;
|
||||
|
||||
fail:
|
||||
close_vgmstream(vgmstream);
|
||||
return NULL;
|
||||
}
|
@ -482,6 +482,7 @@ VGMSTREAM * (*init_vgmstream_functions[])(STREAMFILE *streamFile) = {
|
||||
init_vgmstream_opus_opusnx,
|
||||
init_vgmstream_opus_sqex,
|
||||
init_vgmstream_isb,
|
||||
init_vgmstream_xssb,
|
||||
|
||||
/* lowest priority metas (should go after all metas, and TXTH should go before raw formats) */
|
||||
init_vgmstream_txth, /* proper parsers should supersede TXTH, once added */
|
||||
|
@ -715,6 +715,7 @@ typedef enum {
|
||||
meta_UBI_HX,
|
||||
meta_BMP_KONAMI,
|
||||
meta_ISB,
|
||||
meta_XSSB,
|
||||
|
||||
} meta_t;
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user