#include <assert.h>
#include <math.h>
#include "ao.h"
#include "cpuintrf.h"
#include "aica.h"
#include "aicadsp.h"
static UINT16 PACK(INT32 val)
{
UINT32 temp;
int sign,exponent,k;
sign = (val >> 23) & 0x1;
temp = (val ^ (val << 1)) & 0xFFFFFF;
exponent = 0;
for (k=0; k<12; k++)
{
if (temp & 0x800000)
break;
temp <<= 1;
exponent += 1;
}
if (exponent < 12)
val = (val << exponent) & 0x3FFFFF;
else
val <<= 11;
val >>= 11;
val |= sign << 15;
val |= exponent << 11;
return (UINT16)val;
}
static INT32 UNPACK(UINT16 val)
{
int sign,exponent,mantissa;
INT32 uval;
sign = (val >> 15) & 0x1;
exponent = (val >> 11) & 0xF;
mantissa = val & 0x7FF;
uval = mantissa << 11;
if (exponent > 11)
exponent = 11;
else
uval |= (sign ^ 1) << 22;
uval |= sign << 23;
uval <<= 8;
uval >>= 8;
uval >>= exponent;
return uval;
}
void AICADSP_Init(struct _AICADSP *DSP)
{
memset(DSP,0,sizeof(struct _AICADSP));
DSP->RBL=0x8000;
DSP->Stopped=1;
}
void AICADSP_Step(struct _AICADSP *DSP)
{
INT32 ACC=0; INT32 SHIFTED=0; INT32 X=0; INT32 Y=0; INT32 B=0; INT32 INPUTS=0; INT32 MEMVAL=0;
INT32 FRC_REG=0; INT32 Y_REG=0; UINT32 ADDR=0;
UINT32 ADRS_REG=0; int step;
if(DSP->Stopped)
return;
memset(DSP->EFREG,0,2*16);
#if 0#endif
for(step=0;step<DSP->LastStep;++step)
{
UINT16 *IPtr=DSP->MPRO+step*8;
UINT32 TRA=(IPtr[0]>>9)&0x7F;
UINT32 TWT=(IPtr[0]>>8)&0x01;
UINT32 TWA=(IPtr[0]>>1)&0x7F;
UINT32 XSEL=(IPtr[2]>>15)&0x01;
UINT32 YSEL=(IPtr[2]>>13)&0x03;
UINT32 IRA=(IPtr[2]>>7)&0x3F;
UINT32 IWT=(IPtr[2]>>6)&0x01;
UINT32 IWA=(IPtr[2]>>1)&0x1F;
UINT32 TABLE=(IPtr[4]>>15)&0x01;
UINT32 MWT=(IPtr[4]>>14)&0x01;
UINT32 MRD=(IPtr[4]>>13)&0x01;
UINT32 EWT=(IPtr[4]>>12)&0x01;
UINT32 EWA=(IPtr[4]>>8)&0x0F;
UINT32 ADRL=(IPtr[4]>>7)&0x01;
UINT32 FRCL=(IPtr[4]>>6)&0x01;
UINT32 SHIFT=(IPtr[4]>>4)&0x03;
UINT32 YRL=(IPtr[4]>>3)&0x01;
UINT32 NEGB=(IPtr[4]>>2)&0x01;
UINT32 ZERO=(IPtr[4]>>1)&0x01;
UINT32 BSEL=(IPtr[4]>>0)&0x01;
UINT32 NOFL=(IPtr[6]>>15)&1; UINT32 COEF=step;
UINT32 MASA=(IPtr[6]>>9)&0x3f; UINT32 ADREB=(IPtr[6]>>8)&0x1;
UINT32 NXADR=(IPtr[6]>>7)&0x1;
INT64 v;
#if 0#endif
assert(IRA<0x32);
if(IRA<=0x1f)
INPUTS=DSP->MEMS[IRA];
else if(IRA<=0x2F)
INPUTS=DSP->MIXS[IRA-0x20]<<4; else if(IRA<=0x31)
INPUTS=0;
INPUTS<<=8;
INPUTS>>=8;
if(IWT)
{
DSP->MEMS[IWA]=MEMVAL; if(IRA==IWA)
INPUTS=MEMVAL;
}
if(!ZERO)
{
if(BSEL)
B=ACC;
else
{
B=DSP->TEMP[(TRA+DSP->DEC)&0x7F];
B<<=8;
B>>=8;
}
if(NEGB)
B=0-B;
}
else
B=0;
if(XSEL)
X=INPUTS;
else
{
X=DSP->TEMP[(TRA+DSP->DEC)&0x7F];
X<<=8;
X>>=8;
}
if(YSEL==0)
Y=FRC_REG;
else if(YSEL==1)
Y=DSP->COEF[COEF<<1]>>3; else if(YSEL==2)
Y=(Y_REG>>11)&0x1FFF;
else if(YSEL==3)
Y=(Y_REG>>4)&0x0FFF;
if(YRL)
Y_REG=INPUTS;
if(SHIFT==0)
{
SHIFTED=ACC;
if(SHIFTED>0x007FFFFF)
SHIFTED=0x007FFFFF;
if(SHIFTED<(-0x00800000))
SHIFTED=-0x00800000;
}
else if(SHIFT==1)
{
SHIFTED=ACC*2;
if(SHIFTED>0x007FFFFF)
SHIFTED=0x007FFFFF;
if(SHIFTED<(-0x00800000))
SHIFTED=-0x00800000;
}
else if(SHIFT==2)
{
SHIFTED=ACC*2;
SHIFTED<<=8;
SHIFTED>>=8;
}
else if(SHIFT==3)
{
SHIFTED=ACC;
SHIFTED<<=8;
SHIFTED>>=8;
}
Y<<=19;
Y>>=19;
v=(((INT64) X*(INT64) Y)>>12);
ACC=(int) v+B;
if(TWT)
DSP->TEMP[(TWA+DSP->DEC)&0x7F]=SHIFTED;
if(FRCL)
{
if(SHIFT==3)
FRC_REG=SHIFTED&0x0FFF;
else
FRC_REG=(SHIFTED>>11)&0x1FFF;
}
if(MRD || MWT)
{
ADDR=DSP->MADRS[MASA<<1];
if(!TABLE)
ADDR+=DSP->DEC;
if(ADREB)
ADDR+=ADRS_REG&0x0FFF;
if(NXADR)
ADDR++;
if(!TABLE)
ADDR&=DSP->RBL-1;
else
ADDR&=0xFFFF;
ADDR+=DSP->RBP<<10;
if(MRD && (step&1)) {
if(NOFL)
MEMVAL=DSP->AICARAM[ADDR]<<8;
else
MEMVAL=UNPACK(DSP->AICARAM[ADDR]);
}
if(MWT && (step&1))
{
if(NOFL)
DSP->AICARAM[ADDR]=SHIFTED>>8;
else
DSP->AICARAM[ADDR]=PACK(SHIFTED);
}
}
if(ADRL)
{
if(SHIFT==3)
ADRS_REG=(SHIFTED>>12)&0xFFF;
else
ADRS_REG=(INPUTS>>16);
}
if(EWT)
DSP->EFREG[EWA]+=SHIFTED>>8;
}
--DSP->DEC;
memset(DSP->MIXS,0,4*16);
}
void AICADSP_SetSample(struct _AICADSP *DSP,INT32 sample,int SEL,int MXL)
{
DSP->MIXS[SEL]+=sample;
}
void AICADSP_Start(struct _AICADSP *DSP)
{
int i;
DSP->Stopped=0;
for(i=127;i>=0;--i)
{
UINT16 *IPtr=DSP->MPRO+i*8;
if(IPtr[0]!=0 || IPtr[2]!=0 || IPtr[4]!=0 || IPtr[6]!=0)
break;
}
DSP->LastStep=i+1;
}