#include "meta.h" #include "../coding/coding.h" /* SEG - from Stormfront games [Eragon (multi), Forgotten Realms: Demon Stone (multi) */ VGMSTREAM * init_vgmstream_seg(STREAMFILE *streamFile) { VGMSTREAM * vgmstream = NULL; off_t start_offset; int loop_flag, channel_count; size_t data_size; uint32_t codec; int32_t (*read_32bit)(off_t,STREAMFILE*) = NULL; /* checks */ if (!check_extensions(streamFile, "seg")) goto fail; if (read_32bitBE(0x00,streamFile) != 0x73656700) /* "seg\0" */ goto fail; codec = read_32bitBE(0x04,streamFile); /* 0x08: version? (2: Eragon, Spiderwick Chronicles Wii / 3: Spiderwick Chronicles X360 / 4: Spiderwick Chronicles PC) */ if (guess_endianness32bit(0x08,streamFile)) { read_32bit = read_32bitBE; } else { read_32bit = read_32bitLE; } /* 0x0c: file size */ data_size = read_32bit(0x10, streamFile); /* including interleave padding */ /* 0x14: null */ loop_flag = read_32bit(0x20,streamFile); /* rare */ channel_count = read_32bit(0x24,streamFile); /* 0x28: extradata 1 entries (0x08 per entry, unknown) */ /* 0x2c: extradata 1 offset */ /* 0x30: extradata 2 entries (0x10 or 0x14 per entry, seek/hist table?) */ /* 0x34: extradata 2 offset */ start_offset = 0x4000; /* build the VGMSTREAM */ vgmstream = allocate_vgmstream(channel_count,loop_flag); if (!vgmstream) goto fail; vgmstream->meta_type = meta_SEG; vgmstream->sample_rate = read_32bit(0x18,streamFile); vgmstream->num_samples = read_32bit(0x1c,streamFile); if (loop_flag) { vgmstream->loop_start_sample = 0; vgmstream->loop_end_sample = vgmstream->num_samples; } read_string(vgmstream->stream_name,0x20+1, 0x38,streamFile); switch(codec) { case 0x70733200: /* "ps2\0" */ vgmstream->coding_type = coding_PSX; vgmstream->layout_type = layout_interleave; vgmstream->interleave_block_size = 0x2000; break; case 0x78627800: /* "xbx\0" */ vgmstream->coding_type = coding_XBOX_IMA; vgmstream->layout_type = layout_none; break; case 0x77696900: /* "wii\0" */ vgmstream->coding_type = coding_NGC_DSP; vgmstream->layout_type = layout_interleave; vgmstream->interleave_block_size = 0x2000; /* standard dsp header at start_offset */ dsp_read_coefs_be(vgmstream, streamFile, start_offset+0x1c, vgmstream->interleave_block_size); dsp_read_hist_be(vgmstream, streamFile, start_offset+0x40, vgmstream->interleave_block_size); //todo first_interleave: 0x2000 - 60 break; case 0x70635F00: /* "pc_\0" */ vgmstream->coding_type = coding_IMA; vgmstream->layout_type = layout_none; break; #ifdef VGM_USE_FFMPEG case 0x78623300: { /* "xb3\0" */ uint8_t buf[0x100]; int bytes, block_size, block_count; block_size = 0x4000; block_count = data_size / block_size + (data_size % block_size ? 1 : 0); bytes = ffmpeg_make_riff_xma2(buf,0x100, vgmstream->num_samples, data_size, vgmstream->channels, vgmstream->sample_rate, block_count, block_size); 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; xma_fix_raw_samples(vgmstream, streamFile, start_offset,data_size, 0, 0,0); /* samples are ok */ break; } #endif default: goto fail; } if (!vgmstream_open_stream(vgmstream,streamFile,start_offset)) goto fail; return vgmstream; fail: close_vgmstream(vgmstream); return NULL; }