Updated VGMStream to r1050-1399-gee07eaee.
parent
f58db82054
commit
6661605ea0
|
@ -75,6 +75,8 @@
|
||||||
830EBE1A200465C00023AA10 /* libatrac9.framework in CopyFiles */ = {isa = PBXBuildFile; fileRef = 830EBD9720045F1B0023AA10 /* libatrac9.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; };
|
830EBE1A200465C00023AA10 /* libatrac9.framework in CopyFiles */ = {isa = PBXBuildFile; fileRef = 830EBD9720045F1B0023AA10 /* libatrac9.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; };
|
||||||
8313E3E61902020400B4B6F1 /* mpg123.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 8313E3431901FBDD00B4B6F1 /* mpg123.framework */; };
|
8313E3E61902020400B4B6F1 /* mpg123.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 8313E3431901FBDD00B4B6F1 /* mpg123.framework */; };
|
||||||
8313E3E71902021800B4B6F1 /* mpg123.framework in CopyFiles */ = {isa = PBXBuildFile; fileRef = 8313E3431901FBDD00B4B6F1 /* mpg123.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, ); }; };
|
8313E3E71902021800B4B6F1 /* mpg123.framework in CopyFiles */ = {isa = PBXBuildFile; fileRef = 8313E3431901FBDD00B4B6F1 /* mpg123.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, ); }; };
|
||||||
|
8315958720FEC832007002F0 /* asf_decoder.c in Sources */ = {isa = PBXBuildFile; fileRef = 8315958320FEC831007002F0 /* asf_decoder.c */; };
|
||||||
|
8315958920FEC83F007002F0 /* asf.c in Sources */ = {isa = PBXBuildFile; fileRef = 8315958820FEC83F007002F0 /* asf.c */; };
|
||||||
831BA6181EAC61A500CF89B0 /* adx.c in Sources */ = {isa = PBXBuildFile; fileRef = 831BA60E1EAC61A500CF89B0 /* adx.c */; };
|
831BA6181EAC61A500CF89B0 /* adx.c in Sources */ = {isa = PBXBuildFile; fileRef = 831BA60E1EAC61A500CF89B0 /* adx.c */; };
|
||||||
831BA6191EAC61A500CF89B0 /* ogl.c in Sources */ = {isa = PBXBuildFile; fileRef = 831BA60F1EAC61A500CF89B0 /* ogl.c */; };
|
831BA6191EAC61A500CF89B0 /* ogl.c in Sources */ = {isa = PBXBuildFile; fileRef = 831BA60F1EAC61A500CF89B0 /* ogl.c */; };
|
||||||
831BA61A1EAC61A500CF89B0 /* ps2_vds_vdm.c in Sources */ = {isa = PBXBuildFile; fileRef = 831BA6101EAC61A500CF89B0 /* ps2_vds_vdm.c */; };
|
831BA61A1EAC61A500CF89B0 /* ps2_vds_vdm.c in Sources */ = {isa = PBXBuildFile; fileRef = 831BA6101EAC61A500CF89B0 /* ps2_vds_vdm.c */; };
|
||||||
|
@ -657,6 +659,8 @@
|
||||||
830EBE112004656E0023AA10 /* xnb.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = xnb.c; sourceTree = "<group>"; };
|
830EBE112004656E0023AA10 /* xnb.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = xnb.c; sourceTree = "<group>"; };
|
||||||
830EBE122004656E0023AA10 /* ktss.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = ktss.c; sourceTree = "<group>"; };
|
830EBE122004656E0023AA10 /* ktss.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = ktss.c; sourceTree = "<group>"; };
|
||||||
8313E33D1901FBDC00B4B6F1 /* mpg123.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = mpg123.xcodeproj; path = ../mpg123/mpg123.xcodeproj; sourceTree = "<group>"; };
|
8313E33D1901FBDC00B4B6F1 /* mpg123.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = mpg123.xcodeproj; path = ../mpg123/mpg123.xcodeproj; sourceTree = "<group>"; };
|
||||||
|
8315958320FEC831007002F0 /* asf_decoder.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = asf_decoder.c; sourceTree = "<group>"; };
|
||||||
|
8315958820FEC83F007002F0 /* asf.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = asf.c; sourceTree = "<group>"; };
|
||||||
831BA60E1EAC61A500CF89B0 /* adx.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = adx.c; sourceTree = "<group>"; };
|
831BA60E1EAC61A500CF89B0 /* adx.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = adx.c; sourceTree = "<group>"; };
|
||||||
831BA60F1EAC61A500CF89B0 /* ogl.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = ogl.c; sourceTree = "<group>"; };
|
831BA60F1EAC61A500CF89B0 /* ogl.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = ogl.c; sourceTree = "<group>"; };
|
||||||
831BA6101EAC61A500CF89B0 /* ps2_vds_vdm.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = ps2_vds_vdm.c; sourceTree = "<group>"; };
|
831BA6101EAC61A500CF89B0 /* ps2_vds_vdm.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = ps2_vds_vdm.c; sourceTree = "<group>"; };
|
||||||
|
@ -1199,6 +1203,7 @@
|
||||||
836F6DE018BDC2180095E648 /* acm_decoder.c */,
|
836F6DE018BDC2180095E648 /* acm_decoder.c */,
|
||||||
836F6DE118BDC2180095E648 /* acm_decoder.h */,
|
836F6DE118BDC2180095E648 /* acm_decoder.h */,
|
||||||
836F6DE218BDC2180095E648 /* adx_decoder.c */,
|
836F6DE218BDC2180095E648 /* adx_decoder.c */,
|
||||||
|
8315958320FEC831007002F0 /* asf_decoder.c */,
|
||||||
8306B08120984517000302D4 /* at3plus_decoder.c */,
|
8306B08120984517000302D4 /* at3plus_decoder.c */,
|
||||||
830EBE0F2004655D0023AA10 /* atrac9_decoder.c */,
|
830EBE0F2004655D0023AA10 /* atrac9_decoder.c */,
|
||||||
831BA6221EAC61CB00CF89B0 /* coding_utils.c */,
|
831BA6221EAC61CB00CF89B0 /* coding_utils.c */,
|
||||||
|
@ -1327,6 +1332,7 @@
|
||||||
836F6E3218BDC2180095E648 /* aix.c */,
|
836F6E3218BDC2180095E648 /* aix.c */,
|
||||||
836F6E3318BDC2180095E648 /* akb.c */,
|
836F6E3318BDC2180095E648 /* akb.c */,
|
||||||
836F6E3418BDC2180095E648 /* apple_caff.c */,
|
836F6E3418BDC2180095E648 /* apple_caff.c */,
|
||||||
|
8315958820FEC83F007002F0 /* asf.c */,
|
||||||
836F6E3518BDC2180095E648 /* ast.c */,
|
836F6E3518BDC2180095E648 /* ast.c */,
|
||||||
8306B0D520984590000302D4 /* atsl.c */,
|
8306B0D520984590000302D4 /* atsl.c */,
|
||||||
83A21F7C201D897F000F04B9 /* atx.c */,
|
83A21F7C201D897F000F04B9 /* atx.c */,
|
||||||
|
@ -2052,6 +2058,7 @@
|
||||||
836F6FD518BDC2190095E648 /* ps2_enth.c in Sources */,
|
836F6FD518BDC2190095E648 /* ps2_enth.c in Sources */,
|
||||||
8306B0DC20984590000302D4 /* sthd.c in Sources */,
|
8306B0DC20984590000302D4 /* sthd.c in Sources */,
|
||||||
836F6FAE18BDC2190095E648 /* ngc_dsp_mpds.c in Sources */,
|
836F6FAE18BDC2190095E648 /* ngc_dsp_mpds.c in Sources */,
|
||||||
|
8315958720FEC832007002F0 /* asf_decoder.c in Sources */,
|
||||||
836F705218BDC2190095E648 /* zwdsp.c in Sources */,
|
836F705218BDC2190095E648 /* zwdsp.c in Sources */,
|
||||||
836F6FFB18BDC2190095E648 /* ps2_sps.c in Sources */,
|
836F6FFB18BDC2190095E648 /* ps2_sps.c in Sources */,
|
||||||
836F6FFF18BDC2190095E648 /* ps2_strlr.c in Sources */,
|
836F6FFF18BDC2190095E648 /* ps2_strlr.c in Sources */,
|
||||||
|
@ -2221,6 +2228,7 @@
|
||||||
836F6FE918BDC2190095E648 /* ps2_mihb.c in Sources */,
|
836F6FE918BDC2190095E648 /* ps2_mihb.c in Sources */,
|
||||||
836F6FA918BDC2190095E648 /* ngc_adpdtk.c in Sources */,
|
836F6FA918BDC2190095E648 /* ngc_adpdtk.c in Sources */,
|
||||||
836F6FDC18BDC2190095E648 /* ps2_iab.c in Sources */,
|
836F6FDC18BDC2190095E648 /* ps2_iab.c in Sources */,
|
||||||
|
8315958920FEC83F007002F0 /* asf.c in Sources */,
|
||||||
836F6F3818BDC2190095E648 /* psx_decoder.c in Sources */,
|
836F6F3818BDC2190095E648 /* psx_decoder.c in Sources */,
|
||||||
836F6F2D18BDC2190095E648 /* mpeg_decoder.c in Sources */,
|
836F6F2D18BDC2190095E648 /* mpeg_decoder.c in Sources */,
|
||||||
836F6F2618BDC2190095E648 /* g7221_decoder.c in Sources */,
|
836F6F2618BDC2190095E648 /* g7221_decoder.c in Sources */,
|
||||||
|
|
|
@ -0,0 +1,64 @@
|
||||||
|
#include "coding.h"
|
||||||
|
|
||||||
|
|
||||||
|
/* Decodec Argonaut's ASF ADPCM codec. Algorithm follows Croc2_asf2raw.exe, and the waveform
|
||||||
|
* looks almost correct, but should reverse engineer asfcodec.adl (DLL) for accuracy. */
|
||||||
|
void decode_asf(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do) {
|
||||||
|
off_t frame_offset;
|
||||||
|
int i, frames_in, sample_count = 0;
|
||||||
|
size_t bytes_per_frame, samples_per_frame;
|
||||||
|
uint32_t shift, mode;
|
||||||
|
int32_t hist1 = stream->adpcm_history1_32;
|
||||||
|
int32_t hist2 = stream->adpcm_history2_32;
|
||||||
|
|
||||||
|
/* external interleave (fixed size), mono */
|
||||||
|
bytes_per_frame = 0x11;
|
||||||
|
samples_per_frame = (bytes_per_frame - 0x01) * 2;
|
||||||
|
frames_in = first_sample / samples_per_frame;
|
||||||
|
first_sample = first_sample % samples_per_frame;
|
||||||
|
|
||||||
|
/* parse header */
|
||||||
|
frame_offset = stream->offset + bytes_per_frame*frames_in;
|
||||||
|
shift = (read_8bit(frame_offset+0x00,stream->streamfile) >> 4) & 0xf;
|
||||||
|
mode = (read_8bit(frame_offset+0x00,stream->streamfile) >> 0) & 0xf;
|
||||||
|
|
||||||
|
/* decoder nibbles */
|
||||||
|
for (i = first_sample; i < first_sample + samples_to_do; i++) {
|
||||||
|
int32_t new_sample;
|
||||||
|
uint8_t nibbles = (uint8_t)read_8bit(frame_offset+0x01 + i/2,stream->streamfile);
|
||||||
|
|
||||||
|
new_sample = i&1 ? /* high nibble first */
|
||||||
|
get_low_nibble_signed(nibbles):
|
||||||
|
get_high_nibble_signed(nibbles);
|
||||||
|
new_sample = new_sample << shift;
|
||||||
|
|
||||||
|
switch(mode) {
|
||||||
|
case 0x00:
|
||||||
|
new_sample = new_sample + hist1;
|
||||||
|
//new_sample = (new_sample + (hist1 << 6)) >> 6; /* maybe? */
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 0x04:
|
||||||
|
new_sample = new_sample + hist1*2 - hist2;
|
||||||
|
//new_sample = (new_sample + (hist1 << 7) - (hist2 << 6)) >> 6; /* maybe? */
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: /* other modes (ex 0x02/09) seem only at last frame as 0 */
|
||||||
|
//VGM_LOG("ASF: unknown mode %x at %lx\n", mode,frame_offset);
|
||||||
|
//new_sample = 0; /* maybe? */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
//new_sample = clamp16(new_sample); /* must not */
|
||||||
|
new_sample = new_sample & 0xFFFF; /* probably unnecessary */
|
||||||
|
|
||||||
|
outbuf[sample_count] = new_sample;
|
||||||
|
sample_count += channelspacing;
|
||||||
|
|
||||||
|
hist2 = hist1;
|
||||||
|
hist1 = new_sample;
|
||||||
|
}
|
||||||
|
|
||||||
|
stream->adpcm_history1_32 = hist1;
|
||||||
|
stream->adpcm_history2_32 = hist2;
|
||||||
|
}
|
|
@ -147,6 +147,9 @@ void decode_mc3(VGMSTREAM * vgmstream, VGMSTREAMCHANNEL * stream, sample * outbu
|
||||||
/* fadpcm_decoder */
|
/* fadpcm_decoder */
|
||||||
void decode_fadpcm(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do);
|
void decode_fadpcm(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do);
|
||||||
|
|
||||||
|
/* asf_decoder */
|
||||||
|
void decode_asf(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do);
|
||||||
|
|
||||||
/* ea_mt_decoder*/
|
/* ea_mt_decoder*/
|
||||||
ea_mt_codec_data *init_ea_mt(int channel_count, int type);
|
ea_mt_codec_data *init_ea_mt(int channel_count, int type);
|
||||||
void decode_ea_mt(VGMSTREAM * vgmstream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do, int channel);
|
void decode_ea_mt(VGMSTREAM * vgmstream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do, int channel);
|
||||||
|
|
|
@ -15,21 +15,21 @@ static const int8_t fadpcm_coefs[8][2] = {
|
||||||
|
|
||||||
/* FMOD's FADPCM, basically XA/PSX ADPCM with a fancy header layout.
|
/* FMOD's FADPCM, basically XA/PSX ADPCM with a fancy header layout.
|
||||||
* Code/layout could be simplified but tries to emulate FMOD's code.
|
* Code/layout could be simplified but tries to emulate FMOD's code.
|
||||||
* Algorithm and tables debugged from their PC DLLs. */
|
* Algorithm and tables debugged from their PC DLLs (byte-accurate). */
|
||||||
void decode_fadpcm(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do) {
|
void decode_fadpcm(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacing, int32_t first_sample, int32_t samples_to_do) {
|
||||||
off_t frame_offset;
|
off_t frame_offset;
|
||||||
int i, j, k;
|
int i, j, k;
|
||||||
int block_samples, num_frame, samples_done = 0, sample_count = 0;
|
int block_samples, frames_in, samples_done = 0, sample_count = 0;
|
||||||
uint32_t coefs, shifts;
|
uint32_t coefs, shifts;
|
||||||
int32_t hist1; //= stream->adpcm_history1_32;
|
int32_t hist1; //= stream->adpcm_history1_32;
|
||||||
int32_t hist2; //= stream->adpcm_history2_32;
|
int32_t hist2; //= stream->adpcm_history2_32;
|
||||||
|
|
||||||
/* external interleave (fixed size), mono */
|
/* external interleave (fixed size), mono */
|
||||||
block_samples = (0x8c - 0xc) * 2;
|
block_samples = (0x8c - 0xc) * 2;
|
||||||
num_frame = first_sample / block_samples;
|
frames_in = first_sample / block_samples;
|
||||||
first_sample = first_sample % block_samples;
|
first_sample = first_sample % block_samples;
|
||||||
|
|
||||||
frame_offset = stream->offset + 0x8c*num_frame;
|
frame_offset = stream->offset + 0x8c*frames_in;
|
||||||
|
|
||||||
|
|
||||||
/* parse 0xc header (header samples are not written to outbuf) */
|
/* parse 0xc header (header samples are not written to outbuf) */
|
||||||
|
@ -50,7 +50,7 @@ void decode_fadpcm(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacin
|
||||||
|
|
||||||
coef1 = fadpcm_coefs[coef_index][0];
|
coef1 = fadpcm_coefs[coef_index][0];
|
||||||
coef2 = fadpcm_coefs[coef_index][1];
|
coef2 = fadpcm_coefs[coef_index][1];
|
||||||
shift = 0x16 - shift_factor;
|
shift = 0x16 - shift_factor; /* pre-adjust for 32b sign extend */
|
||||||
|
|
||||||
for (j = 0; j < 4; j++) {
|
for (j = 0; j < 4; j++) {
|
||||||
uint32_t nibbles = read_32bitLE(group_offset + 0x04*j, stream->streamfile);
|
uint32_t nibbles = read_32bitLE(group_offset + 0x04*j, stream->streamfile);
|
||||||
|
@ -59,9 +59,9 @@ void decode_fadpcm(VGMSTREAMCHANNEL * stream, sample * outbuf, int channelspacin
|
||||||
int32_t new_sample;
|
int32_t new_sample;
|
||||||
|
|
||||||
new_sample = (nibbles >> k*4) & 0x0f;
|
new_sample = (nibbles >> k*4) & 0x0f;
|
||||||
new_sample = (new_sample << 28) >> shift; /* sign extend + scale */
|
new_sample = (new_sample << 28) >> shift; /* 32b sign extend + scale */
|
||||||
new_sample = (new_sample - hist2*coef2 + hist1*coef1);
|
new_sample = (new_sample - hist2*coef2 + hist1*coef1);
|
||||||
new_sample = new_sample >> 6; /* (new_sample / 64) has minor rounding differences */
|
new_sample = new_sample >> 6;
|
||||||
new_sample = clamp16(new_sample);
|
new_sample = clamp16(new_sample);
|
||||||
|
|
||||||
if (sample_count >= first_sample && samples_done < samples_to_do) {
|
if (sample_count >= first_sample && samples_done < samples_to_do) {
|
||||||
|
|
|
@ -39,10 +39,16 @@ int ffmpeg_custom_read_eaxma(ffmpeg_codec_data *data, uint8_t *buf, int buf_size
|
||||||
/* read and transform SNS/EA-XMA blocks into XMA packets */
|
/* read and transform SNS/EA-XMA blocks into XMA packets */
|
||||||
while (buf_done < buf_size) {
|
while (buf_done < buf_size) {
|
||||||
int s, p, bytes_to_copy, max_packets;
|
int s, p, bytes_to_copy, max_packets;
|
||||||
size_t data_size = 0, gap_size = 0;
|
size_t block_size, data_size = 0, gap_size = 0;
|
||||||
size_t block_size = read_32bitBE(real_offset, data->streamfile);
|
uint32_t block_flag;
|
||||||
/* 0x04(4): decoded samples */
|
off_t packets_offset;
|
||||||
off_t packets_offset = real_offset + 0x08;
|
|
||||||
|
block_flag = (uint8_t)read_8bit(real_offset+0x00,data->streamfile);
|
||||||
|
block_size = read_32bitBE(real_offset+0x00,data->streamfile) & 0x00FFFFFF;
|
||||||
|
packets_offset = real_offset + 0x08; /* 0x04(4): decoded samples */
|
||||||
|
|
||||||
|
if (block_flag == 0x45) /* exit on last block just in case, though should reach real_size */
|
||||||
|
break;
|
||||||
|
|
||||||
max_packets = get_block_max_packets(num_streams, packets_offset, data->streamfile);
|
max_packets = get_block_max_packets(num_streams, packets_offset, data->streamfile);
|
||||||
if (max_packets == 0) goto fail;
|
if (max_packets == 0) goto fail;
|
||||||
|
@ -115,12 +121,11 @@ int ffmpeg_custom_read_eaxma(ffmpeg_codec_data *data, uint8_t *buf, int buf_size
|
||||||
|
|
||||||
/* move when block is fully done */
|
/* move when block is fully done */
|
||||||
if (data_size == bytes_to_copy + gap_size) {
|
if (data_size == bytes_to_copy + gap_size) {
|
||||||
real_offset += (block_size & 0x00FFFFFF);
|
real_offset += block_size;
|
||||||
virtual_base += data_size;
|
virtual_base += data_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* exit on last block just in case, though should reach real_size */
|
if (block_flag == 0x80) /* exit on last block just in case, though should reach real_size */
|
||||||
if ((block_size & 0x80000000) || (block_size & 0x45000000))
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -153,10 +158,16 @@ int64_t ffmpeg_custom_seek_eaxma(ffmpeg_codec_data *data, int64_t virtual_offset
|
||||||
|
|
||||||
/* find target block */
|
/* find target block */
|
||||||
while (virtual_base < seek_virtual_offset) {
|
while (virtual_base < seek_virtual_offset) {
|
||||||
size_t data_size, extra_size = 0;
|
size_t block_size, data_size, extra_size = 0;
|
||||||
size_t block_size = read_32bitBE(real_offset, data->streamfile);
|
uint32_t block_flag;
|
||||||
|
|
||||||
data_size = (block_size & 0x00FFFFFF) - 0x0c;
|
block_flag = (uint8_t)read_8bit(real_offset+0x00,data->streamfile);
|
||||||
|
block_size = read_32bitBE(real_offset+0x00,data->streamfile) & 0x00FFFFFF;
|
||||||
|
|
||||||
|
if (block_flag == 0x45) /* exit on last block just in case (v1/SPS, empty) */
|
||||||
|
break;
|
||||||
|
|
||||||
|
data_size = block_size - 0x0c;
|
||||||
if (data_size % EAXMA_XMA_PACKET_SIZE)
|
if (data_size % EAXMA_XMA_PACKET_SIZE)
|
||||||
extra_size = EAXMA_XMA_PACKET_SIZE - (data_size % EAXMA_XMA_PACKET_SIZE);
|
extra_size = EAXMA_XMA_PACKET_SIZE - (data_size % EAXMA_XMA_PACKET_SIZE);
|
||||||
|
|
||||||
|
@ -164,8 +175,11 @@ int64_t ffmpeg_custom_seek_eaxma(ffmpeg_codec_data *data, int64_t virtual_offset
|
||||||
if (data_size + extra_size > seek_virtual_offset)
|
if (data_size + extra_size > seek_virtual_offset)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
real_offset += (block_size & 0x00FFFFFF);
|
real_offset += block_size;
|
||||||
virtual_base += data_size + extra_size;
|
virtual_base += data_size + extra_size;
|
||||||
|
|
||||||
|
if (block_flag == 0x80) /* exit on last block just in case (v0/SNS, full) */
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* closest we can use for reads */
|
/* closest we can use for reads */
|
||||||
|
@ -195,13 +209,16 @@ size_t ffmpeg_get_eaxma_virtual_size(int channels, off_t real_offset, size_t rea
|
||||||
/* count all SNS/EAXMA blocks size + padding size */
|
/* count all SNS/EAXMA blocks size + padding size */
|
||||||
while (real_offset < real_end_offset) {
|
while (real_offset < real_end_offset) {
|
||||||
int max_packets;
|
int max_packets;
|
||||||
size_t block_size = read_32bitBE(real_offset + 0x00, streamFile);
|
uint32_t block_flag, block_size;
|
||||||
/* 0x04(4): decoded samples */
|
off_t packets_offset;
|
||||||
off_t packets_offset = real_offset + 0x08;
|
|
||||||
|
block_flag = (uint8_t)read_8bit(real_offset+0x00,streamFile);
|
||||||
|
block_size = read_32bitBE(real_offset+0x00,streamFile) & 0x00FFFFFF;
|
||||||
|
packets_offset = real_offset + 0x08; /* 0x04(4): decoded samples */
|
||||||
|
|
||||||
|
if (block_flag == 0x45) /* exit on last block just in case (v1/SPS, empty) */
|
||||||
|
break;
|
||||||
|
|
||||||
/* At 0x00(1): block flag
|
|
||||||
* - in SNS: 0x00=normal block, 0x80=last block (not mandatory)
|
|
||||||
* - in SPS: 0x48=header, 0x44=normal block, 0x45=last block (empty) */
|
|
||||||
|
|
||||||
max_packets = get_block_max_packets(num_streams, packets_offset, streamFile);
|
max_packets = get_block_max_packets(num_streams, packets_offset, streamFile);
|
||||||
if (max_packets == 0) goto fail;
|
if (max_packets == 0) goto fail;
|
||||||
|
@ -209,10 +226,9 @@ size_t ffmpeg_get_eaxma_virtual_size(int channels, off_t real_offset, size_t rea
|
||||||
/* fixed data_size per block for multichannel, see reads */
|
/* fixed data_size per block for multichannel, see reads */
|
||||||
virtual_size += max_packets * num_streams * EAXMA_XMA_PACKET_SIZE;
|
virtual_size += max_packets * num_streams * EAXMA_XMA_PACKET_SIZE;
|
||||||
|
|
||||||
real_offset += (block_size & 0x00FFFFFF);
|
real_offset += block_size;
|
||||||
|
|
||||||
/* exit on last block just in case, though should reach real_size */
|
if (block_flag == 0x80) /* exit on last block just in case (v0/SNS, full) */
|
||||||
if ((block_size & 0x80000000) || (block_size & 0x45000000))
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,13 +2,13 @@
|
||||||
#include "../util.h"
|
#include "../util.h"
|
||||||
|
|
||||||
// todo this is based on Kazzuya's old code; different emus (PCSX, Mame, Mednafen, etc) do
|
// todo this is based on Kazzuya's old code; different emus (PCSX, Mame, Mednafen, etc) do
|
||||||
// XA coefs int math in different ways (see comments below), so may not be 100% accurate.
|
// XA coefs int math in different ways (see comments below), not be 100% accurate.
|
||||||
// May be implemented like the SNES/SPC700 BRR (per-filter code?).
|
// May be implemented like the SNES/SPC700 BRR (see BSNES' brr.cpp, hardware-tested).
|
||||||
|
|
||||||
/* XA ADPCM gain values */
|
/* XA ADPCM gain values */
|
||||||
static const double K0[4] = { 0.0, 0.9375, 1.796875, 1.53125 };
|
static const double K0[4] = { 0.0, 0.9375, 1.796875, 1.53125 };
|
||||||
static const double K1[4] = { 0.0, 0.0, -0.8125,-0.859375};
|
static const double K1[4] = { 0.0, 0.0, -0.8125,-0.859375};
|
||||||
static int IK0(int fid) { return ((int)((-K0[fid]) * (1 << 10))); } /* K0/1 floats to int, K*2^10 = K*1024 */
|
static int IK0(int fid) { return ((int)((-K0[fid]) * (1 << 10))); } /* K0/1 floats to int, K*2^10 = K*(1<<10) = K*1024 */
|
||||||
static int IK1(int fid) { return ((int)((-K1[fid]) * (1 << 10))); }
|
static int IK1(int fid) { return ((int)((-K1[fid]) * (1 << 10))); }
|
||||||
|
|
||||||
/* Sony XA ADPCM, defined for CD-DA/CD-i in the "Red Book" (private) or "Green Book" (public) specs.
|
/* Sony XA ADPCM, defined for CD-DA/CD-i in the "Red Book" (private) or "Green Book" (public) specs.
|
||||||
|
@ -21,15 +21,16 @@ static int IK1(int fid) { return ((int)((-K1[fid]) * (1 << 10))); }
|
||||||
* short sample = ((nibble << 12) & 0xf000) >> shift
|
* short sample = ((nibble << 12) & 0xf000) >> shift
|
||||||
* or: int sample = ((nibble << 28) & 0xf0000000) >> (shift + N)
|
* or: int sample = ((nibble << 28) & 0xf0000000) >> (shift + N)
|
||||||
* - K0/K1 are float coefs are typically redefined with int math in various ways, with non-equivalent rounding:
|
* - K0/K1 are float coefs are typically redefined with int math in various ways, with non-equivalent rounding:
|
||||||
* (sample + K0*2^N*hist1 + K1*2^N*hist1 + [(2^N)/2]) / 2^N
|
* (sample + K0*2^N*hist1 + K1*2^N*hist2 + [(2^N)/2]) / 2^N
|
||||||
* (sample + K0*2^N*hist1 + K1*2^N*hist1 + [(2^N)/2]) >> N
|
* (sample + K0*2^N*hist1 + K1*2^N*hist2 + [(2^N)/2]) >> N
|
||||||
* sample + (K0<<N*hist1 + K1<<N*hist1)>>N
|
* sample + (K0*2^N*hist1 + K1*2^N*hist2)>>N
|
||||||
* sample + (K0*2^N*hist1)>>N + (K1*2^N*hist1)>>N
|
* sample + (K0*2^N*hist1)>>N + (K1*2^N*hist2)>>N
|
||||||
* etc
|
* etc
|
||||||
* (rounding differences should be inaudible, so public implementations may be approximations)
|
* (rounding differences should be inaudible, so public implementations may be approximations)
|
||||||
*
|
*
|
||||||
* Various XA descendants (PS-ADPCM, EA-XA, NGC DTK, FADPCM, etc) do filters/rounding slightly
|
* Various XA descendants (PS-ADPCM, EA-XA, NGC DTK, FADPCM, etc) do filters/rounding slightly
|
||||||
* differently, using one of the above methods.
|
* differently, using one of the above methods in software/CPU, but in XA's case may be done like
|
||||||
|
* the SNES/SPC700 BRR, with specific per-filter ops.
|
||||||
* int coef tables commonly use N = 6 or 8, so K0 0.9375*64 = 60 or 0.9375*256 = 240
|
* int coef tables commonly use N = 6 or 8, so K0 0.9375*64 = 60 or 0.9375*256 = 240
|
||||||
* PS1 XA is apparently upsampled and interpolated to 44100, vgmstream doesn't simulate this.
|
* PS1 XA is apparently upsampled and interpolated to 44100, vgmstream doesn't simulate this.
|
||||||
*
|
*
|
||||||
|
|
|
@ -547,6 +547,7 @@ static const coding_info coding_info_list[] = {
|
||||||
{coding_MTA2, "Konami MTA2 4-bit ADPCM"},
|
{coding_MTA2, "Konami MTA2 4-bit ADPCM"},
|
||||||
{coding_MC3, "Paradigm MC3 3-bit ADPCM"},
|
{coding_MC3, "Paradigm MC3 3-bit ADPCM"},
|
||||||
{coding_FADPCM, "FMOD FADPCM 4-bit ADPCM"},
|
{coding_FADPCM, "FMOD FADPCM 4-bit ADPCM"},
|
||||||
|
{coding_ASF, "Argonaut ASF 4-bit ADPCM"},
|
||||||
|
|
||||||
{coding_SDX2, "Squareroot-delta-exact (SDX2) 8-bit DPCM"},
|
{coding_SDX2, "Squareroot-delta-exact (SDX2) 8-bit DPCM"},
|
||||||
{coding_SDX2_int, "Squareroot-delta-exact (SDX2) 8-bit DPCM with 1 byte interleave"},
|
{coding_SDX2_int, "Squareroot-delta-exact (SDX2) 8-bit DPCM with 1 byte interleave"},
|
||||||
|
@ -1031,6 +1032,7 @@ static const meta_info meta_info_list[] = {
|
||||||
{meta_DSP_SADF, "Procyon Studio SADF header"},
|
{meta_DSP_SADF, "Procyon Studio SADF header"},
|
||||||
{meta_H4M, "Hudson HVQM4 header"},
|
{meta_H4M, "Hudson HVQM4 header"},
|
||||||
{meta_OGG_MUS, "Ogg Vorbis (MUS header)"},
|
{meta_OGG_MUS, "Ogg Vorbis (MUS header)"},
|
||||||
|
{meta_ASF, "Argonaut ASF header"},
|
||||||
|
|
||||||
#ifdef VGM_USE_FFMPEG
|
#ifdef VGM_USE_FFMPEG
|
||||||
{meta_FFmpeg, "FFmpeg supported file format"},
|
{meta_FFmpeg, "FFmpeg supported file format"},
|
||||||
|
|
|
@ -0,0 +1,56 @@
|
||||||
|
#include "meta.h"
|
||||||
|
#include "../coding/coding.h"
|
||||||
|
|
||||||
|
/* ASF - Argonaut PC games [Croc 2 (PC), Aladdin: Nasira's Revenge (PC)] */
|
||||||
|
VGMSTREAM * init_vgmstream_asf(STREAMFILE *streamFile) {
|
||||||
|
VGMSTREAM * vgmstream = NULL;
|
||||||
|
off_t start_offset;
|
||||||
|
int loop_flag, channel_count, version;
|
||||||
|
|
||||||
|
|
||||||
|
/* checks */
|
||||||
|
if (!check_extensions(streamFile, "asf"))
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
if (read_32bitBE(0x00,streamFile) != 0x41534600) /* "ASF\0" */
|
||||||
|
goto fail;
|
||||||
|
if (read_32bitBE(0x04,streamFile) != 0x02000100)
|
||||||
|
goto fail;
|
||||||
|
if (read_32bitLE(0x08,streamFile) != 0x01 &&
|
||||||
|
read_32bitLE(0x0c,streamFile) != 0x18 &&
|
||||||
|
read_32bitLE(0x1c,streamFile) != 0x20)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
version = read_32bitLE(0x28,streamFile); /* assumed? */
|
||||||
|
switch(version){
|
||||||
|
case 0x0d: channel_count = 1; break; /* Aladdin: Nasira's Revenge (PC) */
|
||||||
|
case 0x0f: channel_count = 2; break; /* Croc 2 (PC), The Emperor's New Groove (PC) */
|
||||||
|
default: goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
loop_flag = 0;
|
||||||
|
start_offset = 0x2c;
|
||||||
|
|
||||||
|
/* build the VGMSTREAM */
|
||||||
|
vgmstream = allocate_vgmstream(channel_count, loop_flag);
|
||||||
|
if (!vgmstream) goto fail;
|
||||||
|
|
||||||
|
vgmstream->sample_rate = (uint16_t)read_16bitLE(0x24, streamFile);
|
||||||
|
vgmstream->meta_type = meta_ASF;
|
||||||
|
vgmstream->coding_type = coding_ASF;
|
||||||
|
vgmstream->layout_type = layout_interleave;
|
||||||
|
vgmstream->interleave_block_size = 0x11;
|
||||||
|
vgmstream->num_samples = (get_streamfile_size(streamFile)-start_offset)/(0x11*channel_count)*32; /* bytes_to_samples */
|
||||||
|
//vgmstream->num_samples = read_32bitLE(0x18,streamFile) * (0x20<<channel_count); /* something like this? */
|
||||||
|
|
||||||
|
read_string(vgmstream->stream_name,0x10, 0x08+1,streamFile);
|
||||||
|
|
||||||
|
|
||||||
|
if (!vgmstream_open_stream(vgmstream,streamFile,start_offset))
|
||||||
|
goto fail;
|
||||||
|
return vgmstream;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
close_vgmstream(vgmstream);
|
||||||
|
return NULL;
|
||||||
|
}
|
|
@ -758,4 +758,6 @@ VGMSTREAM * init_vgmstream_dsp_sadf(STREAMFILE *streamFile);
|
||||||
|
|
||||||
VGMSTREAM * init_vgmstream_h4m(STREAMFILE *streamFile);
|
VGMSTREAM * init_vgmstream_h4m(STREAMFILE *streamFile);
|
||||||
|
|
||||||
|
VGMSTREAM * init_vgmstream_asf(STREAMFILE *streamFile);
|
||||||
|
|
||||||
#endif /*_META_H*/
|
#endif /*_META_H*/
|
||||||
|
|
|
@ -413,6 +413,7 @@ VGMSTREAM * (*init_vgmstream_functions[])(STREAMFILE *streamFile) = {
|
||||||
init_vgmstream_dsp_sadf,
|
init_vgmstream_dsp_sadf,
|
||||||
init_vgmstream_h4m,
|
init_vgmstream_h4m,
|
||||||
init_vgmstream_ps2_ads_container,
|
init_vgmstream_ps2_ads_container,
|
||||||
|
init_vgmstream_asf,
|
||||||
|
|
||||||
init_vgmstream_txth, /* should go at the end (lower priority) */
|
init_vgmstream_txth, /* should go at the end (lower priority) */
|
||||||
#ifdef VGM_USE_FFMPEG
|
#ifdef VGM_USE_FFMPEG
|
||||||
|
@ -1117,6 +1118,8 @@ int get_vgmstream_samples_per_frame(VGMSTREAM * vgmstream) {
|
||||||
return 10;
|
return 10;
|
||||||
case coding_FADPCM:
|
case coding_FADPCM:
|
||||||
return 256; /* (0x8c - 0xc) * 2 */
|
return 256; /* (0x8c - 0xc) * 2 */
|
||||||
|
case coding_ASF:
|
||||||
|
return 32; /* (0x11 - 0x1) * 2 */
|
||||||
case coding_EA_MT:
|
case coding_EA_MT:
|
||||||
return 432;
|
return 432;
|
||||||
case coding_CRI_HCA:
|
case coding_CRI_HCA:
|
||||||
|
@ -1273,6 +1276,8 @@ int get_vgmstream_frame_size(VGMSTREAM * vgmstream) {
|
||||||
return 0x04;
|
return 0x04;
|
||||||
case coding_FADPCM:
|
case coding_FADPCM:
|
||||||
return 0x8c;
|
return 0x8c;
|
||||||
|
case coding_ASF:
|
||||||
|
return 0x11;
|
||||||
case coding_EA_MT:
|
case coding_EA_MT:
|
||||||
return 0; /* variable (frames of bit counts or PCM frames) */
|
return 0; /* variable (frames of bit counts or PCM frames) */
|
||||||
#ifdef VGM_USE_ATRAC9
|
#ifdef VGM_USE_ATRAC9
|
||||||
|
@ -1904,6 +1909,13 @@ void decode_vgmstream(VGMSTREAM * vgmstream, int samples_written, int samples_to
|
||||||
samples_to_do);
|
samples_to_do);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case coding_ASF:
|
||||||
|
for (chan=0;chan<vgmstream->channels;chan++) {
|
||||||
|
decode_asf(&vgmstream->ch[chan],buffer+samples_written*vgmstream->channels+chan,
|
||||||
|
vgmstream->channels,vgmstream->samples_into_block,
|
||||||
|
samples_to_do);
|
||||||
|
}
|
||||||
|
break;
|
||||||
case coding_EA_MT:
|
case coding_EA_MT:
|
||||||
for (chan=0;chan<vgmstream->channels;chan++) {
|
for (chan=0;chan<vgmstream->channels;chan++) {
|
||||||
decode_ea_mt(vgmstream, buffer+samples_written*vgmstream->channels+chan,
|
decode_ea_mt(vgmstream, buffer+samples_written*vgmstream->channels+chan,
|
||||||
|
|
|
@ -162,6 +162,7 @@ typedef enum {
|
||||||
coding_MTA2, /* Konami MTA2 ADPCM */
|
coding_MTA2, /* Konami MTA2 ADPCM */
|
||||||
coding_MC3, /* Paradigm MC3 3-bit ADPCM */
|
coding_MC3, /* Paradigm MC3 3-bit ADPCM */
|
||||||
coding_FADPCM, /* FMOD FADPCM 4-bit ADPCM */
|
coding_FADPCM, /* FMOD FADPCM 4-bit ADPCM */
|
||||||
|
coding_ASF, /* Argonaut ASF 4-bit ADPCM */
|
||||||
|
|
||||||
/* others */
|
/* others */
|
||||||
coding_SDX2, /* SDX2 2:1 Squareroot-Delta-Exact compression DPCM */
|
coding_SDX2, /* SDX2 2:1 Squareroot-Delta-Exact compression DPCM */
|
||||||
|
@ -687,6 +688,7 @@ typedef enum {
|
||||||
meta_OGG_GWM, /* Ogg Vorbis with encryption [Metronomicon (PC)] */
|
meta_OGG_GWM, /* Ogg Vorbis with encryption [Metronomicon (PC)] */
|
||||||
meta_H4M, /* Hudson HVQM4 video [Resident Evil 0 (GC), Tales of Symphonia (GC)] */
|
meta_H4M, /* Hudson HVQM4 video [Resident Evil 0 (GC), Tales of Symphonia (GC)] */
|
||||||
meta_OGG_MUS, /* Ogg Vorbis with encryption [Redux - Dark Matters (PC)] */
|
meta_OGG_MUS, /* Ogg Vorbis with encryption [Redux - Dark Matters (PC)] */
|
||||||
|
meta_ASF, /* Argonaut ASF [Croc 2 (PC)] */
|
||||||
|
|
||||||
#ifdef VGM_USE_FFMPEG
|
#ifdef VGM_USE_FFMPEG
|
||||||
meta_FFmpeg,
|
meta_FFmpeg,
|
||||||
|
|
Loading…
Reference in New Issue