#define _IN_SPU
#include "../peops/stdafx.h"
#include "../peops/externals.h"
#include "../peops/regs.h"
#include "../peops/registers.h"
#include "../peops/spu.h"
void SPUirq(void) ;
static u16 regArea[0x200];
static u16 spuMem[256*1024];
static u8 * spuMemC;
static u8 * pSpuIrq=0;
static u8 * pSpuBuffer;
static int iVolume;
static SPUCHAN s_chan[MAXCHAN+1]; static REVERBInfo rvb;
static u32 dwNoiseVal=1;
static u16 spuCtrl=0; static u16 spuStat=0;
static u16 spuIrq=0;
static u32 spuAddr=0xffffffff; static int bSPUIsOpen=0;
static const int f[5][2] = {
{ 0, 0 },
{ 60, 0 },
{ 115, -52 },
{ 98, -55 },
{ 122, -60 } };
static s16 * pS;
static s32 ttemp;
#include "../peops/reverb.c"
#include "../peops/adsr.c"
#include "../peops/registers.c"
#include "../peops/dma.c"
#define gval0 (((int *)(&s_chan[ch].SB[29]))[gpos])
#define gval(x) (((int *)(&s_chan[ch].SB[29]))[(gpos+x)&3])
#include "gauss_i.h"
static INLINE void StartSound(int ch)
{
StartADSR(ch);
s_chan[ch].pCurr=s_chan[ch].pStart;
s_chan[ch].s_1=0; s_chan[ch].s_2=0;
s_chan[ch].iSBPos=28;
s_chan[ch].bNew=0; s_chan[ch].bStop=0;
s_chan[ch].bOn=1;
s_chan[ch].SB[29]=0; s_chan[ch].SB[30]=0;
s_chan[ch].spos=0x40000L;s_chan[ch].SB[28]=0; }
static u32 sampcount;
static u32 decaybegin;
static u32 decayend;
void setlength(s32 stop, s32 fade)
{
if(stop==~0)
{
decaybegin=~0;
}
else
{
stop=(stop*441)/10;
fade=(fade*441)/10;
decaybegin=stop;
decayend=stop+fade;
}
}
#define CLIP(_x) {if(_x>32767) _x=32767; if(_x<-32767) _x=-32767;}
int SPUasync(u32 cycles)
{
int volmul=iVolume;
static s32 dosampies;
s32 temp;
ttemp+=cycles;
dosampies=ttemp/384;
if(!dosampies) return(1);
ttemp-=dosampies*384;
temp=dosampies;
while(temp)
{
s32 revLeft=0, revRight=0;
s32 sl=0, sr=0;
int ch,fa;
temp--;
{
for(ch=0;ch<MAXCHAN;ch++) {
if(s_chan[ch].bNew) StartSound(ch); if(!s_chan[ch].bOn) continue;
if(s_chan[ch].iActFreq!=s_chan[ch].iUsedFreq) {
s_chan[ch].iUsedFreq=s_chan[ch].iActFreq; s_chan[ch].sinc=s_chan[ch].iRawPitch<<4;
if(!s_chan[ch].sinc) s_chan[ch].sinc=1;
}
while(s_chan[ch].spos>=0x10000L)
{
if(s_chan[ch].iSBPos==28) {
int predict_nr,shift_factor,flags,d,s;
u8* start;unsigned int nSample;
int s_1,s_2;
start=s_chan[ch].pCurr;
if (start == (u8*)-1) {
s_chan[ch].bOn=0; s_chan[ch].ADSRX.lVolume=0;
s_chan[ch].ADSRX.EnvelopeVol=0;
goto ENDX; }
s_chan[ch].iSBPos=0;
s_1=s_chan[ch].s_1;
s_2=s_chan[ch].s_2;
predict_nr=(int)*start;start++;
shift_factor=predict_nr&0xf;
predict_nr >>= 4;
flags=(int)*start;start++;
for (nSample=0;nSample<28;start++)
{
d=(int)*start;
s=((d&0xf)<<12);
if(s&0x8000) s|=0xffff0000;
fa=(s >> shift_factor);
fa=fa + ((s_1 * f[predict_nr][0])>>6) + ((s_2 * f[predict_nr][1])>>6);
s_2=s_1;s_1=fa;
s=((d & 0xf0) << 8);
s_chan[ch].SB[nSample++]=fa;
if(s&0x8000) s|=0xffff0000;
fa=(s>>shift_factor);
fa=fa + ((s_1 * f[predict_nr][0])>>6) + ((s_2 * f[predict_nr][1])>>6);
s_2=s_1;s_1=fa;
s_chan[ch].SB[nSample++]=fa;
}
if(spuCtrl&0x40) {
if((pSpuIrq > start-16 && pSpuIrq <= start) ||
((flags&1) && (pSpuIrq > s_chan[ch].pLoop-16 &&
pSpuIrq <= s_chan[ch].pLoop)))
{
s_chan[ch].iIrqDone=1; SPUirq();
}
}
if((flags&4) && (!s_chan[ch].bIgnoreLoop))
s_chan[ch].pLoop=start-16;
if(flags&1) {
if(flags!=3 || s_chan[ch].pLoop==NULL) { start = (u8*)-1;
}
else
{
start = s_chan[ch].pLoop;
}
}
s_chan[ch].pCurr=start; s_chan[ch].s_1=s_1;
s_chan[ch].s_2=s_2;
}
fa=s_chan[ch].SB[s_chan[ch].iSBPos++];
if((spuCtrl&0x4000)==0) fa=0; else CLIP(fa);
{
int gpos;
gpos = s_chan[ch].SB[28];
gval0 = fa;
gpos = (gpos+1) & 3;
s_chan[ch].SB[28] = gpos;
}
s_chan[ch].spos -= 0x10000L;
}
if(s_chan[ch].bNoise)
{
if((dwNoiseVal<<=1)&0x80000000L)
{
dwNoiseVal^=0x0040001L;
fa=((dwNoiseVal>>2)&0x7fff);
fa=-fa;
}
else fa=(dwNoiseVal>>2)&0x7fff;
fa=s_chan[ch].iOldNoise+((fa-s_chan[ch].iOldNoise)/((0x001f-((spuCtrl&0x3f00)>>9))+1));
if(fa>32767L) fa=32767L;
if(fa<-32767L) fa=-32767L;
s_chan[ch].iOldNoise=fa;
} else {
int vl, vr, gpos;
vl = (s_chan[ch].spos >> 6) & ~3;
gpos = s_chan[ch].SB[28];
vr=(gauss[vl]*gval0)>>9;
vr+=(gauss[vl+1]*gval(1))>>9;
vr+=(gauss[vl+2]*gval(2))>>9;
vr+=(gauss[vl+3]*gval(3))>>9;
fa = vr>>2;
}
s_chan[ch].sval = (MixADSR(ch) * fa)>>10; if(s_chan[ch].bFMod==2) {
int NP=s_chan[ch+1].iRawPitch;
NP=((32768L+s_chan[ch].sval)*NP)>>15;
if(NP>0x3fff) NP=0x3fff;
if(NP<0x1) NP=0x1;
NP=(44100L*NP)/(4096L);
s_chan[ch+1].iActFreq=NP;
s_chan[ch+1].iUsedFreq=NP;
s_chan[ch+1].sinc=(((NP/10)<<16)/4410);
if(!s_chan[ch+1].sinc) s_chan[ch+1].sinc=1;
}
else
{
int tmpl,tmpr;
if (1) {
tmpl=(s_chan[ch].sval*s_chan[ch].iLeftVolume)>>14;
tmpr=(s_chan[ch].sval*s_chan[ch].iRightVolume)>>14;
} else {
tmpl = 0;
tmpr = 0;
}
sl+=tmpl;
sr+=tmpr;
if(((rvb.Enabled>>ch)&1) && (spuCtrl&0x80))
{
revLeft+=tmpl;
revRight+=tmpr;
}
}
s_chan[ch].spos += s_chan[ch].sinc;
ENDX: ;
}
}
MixREVERBLeftRight(&sl,&sr,revLeft,revRight);
if(sampcount>=decaybegin)
{
s32 dmul;
if(decaybegin!=~0) {
if(sampcount>=decayend)
{
return(0);
}
dmul=256-(256*(sampcount-decaybegin)/(decayend-decaybegin));
sl=(sl*dmul)>>8;
sr=(sr*dmul)>>8;
}
}
sampcount++;
sl=(sl*volmul)>>8;
sr=(sr*volmul)>>8;
if(sl>32767) sl=32767; if(sl<-32767) sl=-32767;
if(sr>32767) sr=32767; if(sr<-32767) sr=-32767;
*pS++=sl;
*pS++=sr;
}
return(1);
}
void SPU_flushboot(void)
{
if((u8*)pS>((u8*)pSpuBuffer+1024))
{
spu_update((u8*)pSpuBuffer,(u8*)pS-(u8*)pSpuBuffer);
pS=(s16 *)pSpuBuffer;
}
}
#ifdef TIMEO
static u64 begintime;
static u64 gettime64(void)
{
struct timeval tv;
u64 ret;
gettimeofday(&tv,0);
ret=tv.tv_sec;
ret*=1000000;
ret+=tv.tv_usec;
return(ret);
}
#endif
int SPUinit(void)
{
spuMemC=(u8*)spuMem; memset((void *)s_chan,0,MAXCHAN*sizeof(SPUCHAN));
memset((void *)&rvb,0,sizeof(REVERBInfo));
memset(regArea,0,sizeof(regArea));
memset(spuMem,0,sizeof(spuMem));
InitADSR();
sampcount=ttemp=0;
#ifdef TIMEO
begintime=gettime64();
#endif
return 0;
}
void SetupStreams(void)
{
int i;
pSpuBuffer=(u8*)malloc(32768); pS=(s16 *)pSpuBuffer;
for(i=0;i<MAXCHAN;i++) {
s_chan[i].ADSRX.SustainLevel = 1024; s_chan[i].iIrqDone=0;
s_chan[i].pLoop=spuMemC;
s_chan[i].pStart=spuMemC;
s_chan[i].pCurr=spuMemC;
}
}
void RemoveStreams(void)
{
free(pSpuBuffer); pSpuBuffer=NULL;
#ifdef TIMEO
{
u64 tmp;
tmp=gettime64();
tmp-=begintime;
if(tmp)
tmp=(u64)sampcount*1000000/tmp;
printf("%lld samples per second\n",tmp);
}
#endif
}
int SPUopen(void)
{
if(bSPUIsOpen) return 0; spuIrq=0;
spuStat=spuCtrl=0;
spuAddr=0xffffffff;
dwNoiseVal=1;
spuMemC=(u8*)spuMem;
memset((void *)s_chan,0,(MAXCHAN+1)*sizeof(SPUCHAN));
pSpuIrq=0;
iVolume=255; SetupStreams();
bSPUIsOpen=1;
return 1;
}
int SPUclose(void)
{
if(!bSPUIsOpen) return 0;
bSPUIsOpen=0;
RemoveStreams();
return 0;
}
int SPUshutdown(void)
{
return 0;
}
void SPUinjectRAMImage(u16 *pIncoming)
{
int i;
for (i = 0; i < (256*1024); i++)
{
spuMem[i] = pIncoming[i];
}
}