2008-04-03 15:56:50 +02:00
|
|
|
#ifdef _MSC_VER
|
2008-04-03 15:40:36 +02:00
|
|
|
#define _USE_MATH_DEFINES
|
|
|
|
#endif
|
2008-01-31 07:04:26 +01:00
|
|
|
#include <math.h>
|
2008-05-06 05:35:37 +02:00
|
|
|
#include "meta.h"
|
2008-01-31 07:04:26 +01:00
|
|
|
#include "../util.h"
|
|
|
|
|
2008-05-20 17:18:38 +02:00
|
|
|
VGMSTREAM * init_vgmstream_adx(STREAMFILE *streamFile) {
|
2008-01-31 07:04:26 +01:00
|
|
|
VGMSTREAM * vgmstream = NULL;
|
2008-05-10 05:56:39 +02:00
|
|
|
off_t stream_offset;
|
2008-01-31 07:04:26 +01:00
|
|
|
size_t filesize;
|
2008-05-10 05:56:39 +02:00
|
|
|
uint16_t version_signature;
|
2008-01-31 07:04:26 +01:00
|
|
|
int loop_flag=0;
|
|
|
|
int channel_count;
|
2008-02-15 19:49:59 +01:00
|
|
|
int32_t loop_start_offset=0;
|
|
|
|
int32_t loop_end_offset=0;
|
|
|
|
int32_t loop_start_sample=0;
|
|
|
|
int32_t loop_end_sample=0;
|
2008-01-31 07:04:26 +01:00
|
|
|
meta_t header_type;
|
|
|
|
int16_t coef1, coef2;
|
2008-05-10 05:56:39 +02:00
|
|
|
uint16_t cutoff;
|
2008-05-20 17:18:38 +02:00
|
|
|
char filename[260];
|
2008-01-31 07:04:26 +01:00
|
|
|
|
|
|
|
/* check extension, case insensitive */
|
2008-05-20 17:18:38 +02:00
|
|
|
streamFile->get_name(streamFile,filename,sizeof(filename));
|
2008-01-31 07:04:26 +01:00
|
|
|
if (strcasecmp("adx",filename_extension(filename))) goto fail;
|
|
|
|
|
2008-05-20 17:18:38 +02:00
|
|
|
filesize = get_streamfile_size(streamFile);
|
2008-01-31 07:04:26 +01:00
|
|
|
|
|
|
|
/* check first 2 bytes */
|
2008-05-20 17:18:38 +02:00
|
|
|
if ((uint16_t)read_16bitBE(0,streamFile)!=0x8000) goto fail;
|
2008-01-31 07:04:26 +01:00
|
|
|
|
2008-05-10 05:56:39 +02:00
|
|
|
/* get stream offset, check for CRI signature just before */
|
2008-05-20 17:18:38 +02:00
|
|
|
stream_offset = (uint16_t)read_16bitBE(2,streamFile) + 4;
|
|
|
|
if ((uint16_t)read_16bitBE(stream_offset-6,streamFile)!=0x2863 ||/* "(c" */
|
|
|
|
(uint32_t)read_32bitBE(stream_offset-4,streamFile)!=0x29435249 /* ")CRI" */
|
2008-05-10 05:56:39 +02:00
|
|
|
) goto fail;
|
|
|
|
|
|
|
|
/* check for encoding type */
|
|
|
|
/* 2 is for some unknown fixed filter, 3 is standard ADX, 4 is
|
|
|
|
* ADX with exponential scale, 0x11 is AHX */
|
2008-05-20 17:18:38 +02:00
|
|
|
if (read_8bit(4,streamFile) != 3) goto fail;
|
2008-05-10 05:56:39 +02:00
|
|
|
|
|
|
|
/* check for frame size (only 18 is supported at the moment) */
|
2008-05-20 17:18:38 +02:00
|
|
|
if (read_8bit(5,streamFile) != 18) goto fail;
|
2008-05-10 05:56:39 +02:00
|
|
|
|
|
|
|
/* check for bits per sample? (only 4 makes sense for ADX) */
|
2008-05-20 17:18:38 +02:00
|
|
|
if (read_8bit(6,streamFile) != 4) goto fail;
|
2008-01-31 07:04:26 +01:00
|
|
|
|
|
|
|
/* check version signature, read loop info */
|
2008-05-20 17:18:38 +02:00
|
|
|
version_signature = read_16bitBE(0x12,streamFile);
|
2008-05-10 05:56:39 +02:00
|
|
|
if (version_signature == 0x0300) { /* type 03 */
|
2008-01-31 07:04:26 +01:00
|
|
|
header_type = meta_ADX_03;
|
2008-05-10 05:56:39 +02:00
|
|
|
if (stream_offset-6 >= 0x2c) { /* enough space for loop info? */
|
2008-05-20 17:18:38 +02:00
|
|
|
loop_flag = (read_32bitBE(0x18,streamFile) != 0);
|
|
|
|
loop_start_sample = read_32bitBE(0x1c,streamFile);
|
|
|
|
loop_start_offset = read_32bitBE(0x20,streamFile);
|
|
|
|
loop_end_sample = read_32bitBE(0x24,streamFile);
|
|
|
|
loop_end_offset = read_32bitBE(0x28,streamFile);
|
2008-01-31 07:04:26 +01:00
|
|
|
}
|
2008-05-10 05:56:39 +02:00
|
|
|
} else if (version_signature == 0x0400) {
|
2008-05-03 21:41:28 +02:00
|
|
|
|
2008-05-10 05:56:39 +02:00
|
|
|
off_t ainf_info_length=0;
|
2008-05-03 21:41:28 +02:00
|
|
|
|
2008-05-20 17:18:38 +02:00
|
|
|
if((uint32_t)read_32bitBE(0x24,streamFile)==0x41494E46) /* AINF Header */
|
|
|
|
ainf_info_length = (off_t)read_32bitBE(0x28,streamFile);
|
2008-05-03 21:41:28 +02:00
|
|
|
|
2008-05-10 05:56:39 +02:00
|
|
|
header_type = meta_ADX_04;
|
|
|
|
if (stream_offset-ainf_info_length-6 >= 0x38) { /* enough space for loop info? */
|
2008-05-20 17:18:38 +02:00
|
|
|
loop_flag = (read_32bitBE(0x24,streamFile) != 0);
|
|
|
|
loop_start_sample = read_32bitBE(0x28,streamFile);
|
|
|
|
loop_start_offset = read_32bitBE(0x2c,streamFile);
|
|
|
|
loop_end_sample = read_32bitBE(0x30,streamFile);
|
|
|
|
loop_end_offset = read_32bitBE(0x34,streamFile);
|
2008-01-31 07:04:26 +01:00
|
|
|
}
|
2008-05-10 05:56:39 +02:00
|
|
|
} else if (version_signature == 0x0500) { /* found in some SFD : Buggy Heat, appears to have no loop */
|
|
|
|
header_type = meta_ADX_05;
|
2008-01-31 07:04:26 +01:00
|
|
|
} else goto fail; /* not a known/supported version signature */
|
|
|
|
|
|
|
|
/* At this point we almost certainly have an ADX file,
|
|
|
|
* so let's build the VGMSTREAM. */
|
|
|
|
|
2008-05-10 05:56:39 +02:00
|
|
|
/* high-pass cutoff frequency, always 500 that I've seen */
|
2008-05-20 17:18:38 +02:00
|
|
|
cutoff = (uint16_t)read_16bitBE(0x10,streamFile);
|
2008-05-10 05:56:39 +02:00
|
|
|
|
2008-05-20 17:18:38 +02:00
|
|
|
channel_count = read_8bit(7,streamFile);
|
2008-01-31 07:04:26 +01:00
|
|
|
vgmstream = allocate_vgmstream(channel_count,loop_flag);
|
|
|
|
if (!vgmstream) goto fail;
|
|
|
|
|
|
|
|
/* fill in the vital statistics */
|
2008-05-20 17:18:38 +02:00
|
|
|
vgmstream->num_samples = read_32bitBE(0xc,streamFile);
|
|
|
|
vgmstream->sample_rate = read_32bitBE(8,streamFile);
|
2008-01-31 07:04:26 +01:00
|
|
|
/* channels and loop flag are set by allocate_vgmstream */
|
|
|
|
vgmstream->loop_start_sample = loop_start_sample;
|
|
|
|
vgmstream->loop_end_sample = loop_end_sample;
|
|
|
|
|
|
|
|
vgmstream->coding_type = coding_CRI_ADX;
|
2008-02-05 01:03:39 +01:00
|
|
|
if (channel_count==1)
|
|
|
|
vgmstream->layout_type = layout_none;
|
|
|
|
else
|
|
|
|
vgmstream->layout_type = layout_interleave;
|
2008-01-31 07:04:26 +01:00
|
|
|
vgmstream->meta_type = header_type;
|
|
|
|
|
|
|
|
vgmstream->interleave_block_size=18;
|
|
|
|
|
|
|
|
/* calculate filter coefficients */
|
|
|
|
{
|
|
|
|
double x,y,z,a,b,c;
|
|
|
|
|
2008-05-10 05:56:39 +02:00
|
|
|
x = cutoff;
|
2008-01-31 07:04:26 +01:00
|
|
|
y = vgmstream->sample_rate;
|
|
|
|
z = cos(2.0*M_PI*x/y);
|
|
|
|
|
|
|
|
a = M_SQRT2-z;
|
|
|
|
b = M_SQRT2-1.0;
|
|
|
|
c = (a-sqrt((a+b)*(a-b)))/b;
|
|
|
|
|
|
|
|
coef1 = floor(c*8192);
|
|
|
|
coef2 = floor(c*c*-4096);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
int i;
|
2008-05-20 22:19:46 +02:00
|
|
|
STREAMFILE * chstreamfile;
|
|
|
|
|
|
|
|
/* ADX is so tightly interleaved that having two buffers is silly */
|
|
|
|
chstreamfile = streamFile->open(streamFile,filename,18*0x400);
|
|
|
|
if (!chstreamfile) goto fail;
|
|
|
|
|
2008-01-31 07:04:26 +01:00
|
|
|
for (i=0;i<channel_count;i++) {
|
2008-05-20 22:19:46 +02:00
|
|
|
vgmstream->ch[i].streamfile = chstreamfile;
|
2008-01-31 07:04:26 +01:00
|
|
|
|
|
|
|
vgmstream->ch[i].channel_start_offset=
|
|
|
|
vgmstream->ch[i].offset=
|
2008-05-10 05:56:39 +02:00
|
|
|
stream_offset+18*i;
|
2008-01-31 07:04:26 +01:00
|
|
|
|
|
|
|
vgmstream->ch[i].adpcm_coef[0] = coef1;
|
|
|
|
vgmstream->ch[i].adpcm_coef[1] = coef2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return vgmstream;
|
|
|
|
|
|
|
|
/* clean up anything we may have opened */
|
|
|
|
fail:
|
|
|
|
if (vgmstream) close_vgmstream(vgmstream);
|
|
|
|
return NULL;
|
|
|
|
}
|