diff options
author | Alexey Yakovenko <wakeroid@gmail.com> | 2010-06-22 22:26:45 +0200 |
---|---|---|
committer | Alexey Yakovenko <wakeroid@gmail.com> | 2010-06-22 22:26:45 +0200 |
commit | 20575a338e9640eca924958484a5fee800e09971 (patch) | |
tree | efaeac41a8a21bea1c90494b3b0968169d4f8732 /plugins/ao/eng_ssf/scsp.c | |
parent | c901a7cbf52ee234220f21b85c2e77667264d16c (diff) |
audio overload plugin - highly experimental; no seeking; crashes
Diffstat (limited to 'plugins/ao/eng_ssf/scsp.c')
-rw-r--r-- | plugins/ao/eng_ssf/scsp.c | 1268 |
1 files changed, 1268 insertions, 0 deletions
diff --git a/plugins/ao/eng_ssf/scsp.c b/plugins/ao/eng_ssf/scsp.c new file mode 100644 index 00000000..a4ed2d6a --- /dev/null +++ b/plugins/ao/eng_ssf/scsp.c @@ -0,0 +1,1268 @@ +/* + Sega/Yamaha YMF292-F (SCSP = Saturn Custom Sound Processor) emulation + By ElSemi + MAME/M1 conversion and cleanup by R. Belmont + Additional code and bugfixes by kingshriek + + This chip has 32 voices. Each voice can play a sample or be part of + an FM construct. Unlike traditional Yamaha FM chips, the base waveform + for the FM still comes from the wavetable RAM. + + ChangeLog: + * November 25, 2003 (ES) Fixed buggy timers and envelope overflows. + (RB) Improved sample rates other than 44100, multiple + chips now works properly. + * December 02, 2003 (ES) Added DISDL register support, improves mix. + * April 28, 2004 (ES) Corrected envelope rates, added key-rate scaling, + added ringbuffer support. + * January 8, 2005 (RB) Added ability to specify region offset for RAM. + * January 26, 2007 (ES) Added on-board DSP capability + * December 16, 2007 (kingshriek) Many EG bug fixes, implemented effects mixer, + implemented FM. +*/ + +#include <math.h> +#include <string.h> +#include "ao.h" +#include "cpuintrf.h" +#include "scsp.h" +#include "scspdsp.h" +#include "sat_hw.h" + +#define ICLIP16(x) (x<-32768)?-32768:((x>32767)?32767:x) + +#define SHIFT 12 +#define FIX(v) ((UINT32) ((float) (1<<SHIFT)*(v))) + + +#define EG_SHIFT 16 +#define FM_DELAY 0 // delay in number of slots processed before samples are written to the FM ring buffer + +// include the LFO handling code +#include "scsplfo.c" + +/* + SCSP features 32 programmable slots + that can generate FM and PCM (from ROM/RAM) sound +*/ + +//SLOT PARAMETERS +#define KEYONEX(slot) ((slot->udata.data[0x0]>>0x0)&0x1000) +#define KEYONB(slot) ((slot->udata.data[0x0]>>0x0)&0x0800) +#define SBCTL(slot) ((slot->udata.data[0x0]>>0x9)&0x0003) +#define SSCTL(slot) ((slot->udata.data[0x0]>>0x7)&0x0003) +#define LPCTL(slot) ((slot->udata.data[0x0]>>0x5)&0x0003) +#define PCM8B(slot) ((slot->udata.data[0x0]>>0x0)&0x0010) + +#define SA(slot) (((slot->udata.data[0x0]&0xF)<<16)|(slot->udata.data[0x1])) + +#define LSA(slot) (slot->udata.data[0x2]) + +#define LEA(slot) (slot->udata.data[0x3]) + +#define D2R(slot) ((slot->udata.data[0x4]>>0xB)&0x001F) +#define D1R(slot) ((slot->udata.data[0x4]>>0x6)&0x001F) +#define EGHOLD(slot) ((slot->udata.data[0x4]>>0x0)&0x0020) +#define AR(slot) ((slot->udata.data[0x4]>>0x0)&0x001F) + +#define LPSLNK(slot) ((slot->udata.data[0x5]>>0x0)&0x4000) +#define KRS(slot) ((slot->udata.data[0x5]>>0xA)&0x000F) +#define DL(slot) ((slot->udata.data[0x5]>>0x5)&0x001F) +#define RR(slot) ((slot->udata.data[0x5]>>0x0)&0x001F) + +#define STWINH(slot) ((slot->udata.data[0x6]>>0x0)&0x0200) +#define SDIR(slot) ((slot->udata.data[0x6]>>0x0)&0x0100) +#define TL(slot) ((slot->udata.data[0x6]>>0x0)&0x00FF) + +#define MDL(slot) ((slot->udata.data[0x7]>>0xC)&0x000F) +#define MDXSL(slot) ((slot->udata.data[0x7]>>0x6)&0x003F) +#define MDYSL(slot) ((slot->udata.data[0x7]>>0x0)&0x003F) + +#define OCT(slot) ((slot->udata.data[0x8]>>0xB)&0x000F) +#define FNS(slot) ((slot->udata.data[0x8]>>0x0)&0x03FF) + +#define LFORE(slot) ((slot->udata.data[0x9]>>0x0)&0x8000) +#define LFOF(slot) ((slot->udata.data[0x9]>>0xA)&0x001F) +#define PLFOWS(slot) ((slot->udata.data[0x9]>>0x8)&0x0003) +#define PLFOS(slot) ((slot->udata.data[0x9]>>0x5)&0x0007) +#define ALFOWS(slot) ((slot->udata.data[0x9]>>0x3)&0x0003) +#define ALFOS(slot) ((slot->udata.data[0x9]>>0x0)&0x0007) + +#define ISEL(slot) ((slot->udata.data[0xA]>>0x3)&0x000F) +#define IMXL(slot) ((slot->udata.data[0xA]>>0x0)&0x0007) + +#define DISDL(slot) ((slot->udata.data[0xB]>>0xD)&0x0007) +#define DIPAN(slot) ((slot->udata.data[0xB]>>0x8)&0x001F) +#define EFSDL(slot) ((slot->udata.data[0xB]>>0x5)&0x0007) +#define EFPAN(slot) ((slot->udata.data[0xB]>>0x0)&0x001F) + +//Envelope times in ms +static const double ARTimes[64]={100000/*infinity*/,100000/*infinity*/,8100.0,6900.0,6000.0,4800.0,4000.0,3400.0,3000.0,2400.0,2000.0,1700.0,1500.0, + 1200.0,1000.0,860.0,760.0,600.0,500.0,430.0,380.0,300.0,250.0,220.0,190.0,150.0,130.0,110.0,95.0, + 76.0,63.0,55.0,47.0,38.0,31.0,27.0,24.0,19.0,15.0,13.0,12.0,9.4,7.9,6.8,6.0,4.7,3.8,3.4,3.0,2.4, + 2.0,1.8,1.6,1.3,1.1,0.93,0.85,0.65,0.53,0.44,0.40,0.35,0.0,0.0}; +static const double DRTimes[64]={100000/*infinity*/,100000/*infinity*/,118200.0,101300.0,88600.0,70900.0,59100.0,50700.0,44300.0,35500.0,29600.0,25300.0,22200.0,17700.0, + 14800.0,12700.0,11100.0,8900.0,7400.0,6300.0,5500.0,4400.0,3700.0,3200.0,2800.0,2200.0,1800.0,1600.0,1400.0,1100.0, + 920.0,790.0,690.0,550.0,460.0,390.0,340.0,270.0,230.0,200.0,170.0,140.0,110.0,98.0,85.0,68.0,57.0,49.0,43.0,34.0, + 28.0,25.0,22.0,18.0,14.0,12.0,11.0,8.5,7.1,6.1,5.4,4.3,3.6,3.1}; +static UINT32 FNS_Table[0x400]; +static INT32 EG_TABLE[0x400]; + +typedef enum {ATTACK,DECAY1,DECAY2,RELEASE} _STATE; +struct _EG +{ + int volume; // + _STATE state; + int step; + //step vals + int AR; //Attack + int D1R; //Decay1 + int D2R; //Decay2 + int RR; //Release + + int DL; //Decay level + UINT8 EGHOLD; + UINT8 LPLINK; +}; + +struct _SLOT +{ + union + { + UINT16 data[0x10]; //only 0x1a bytes used + UINT8 datab[0x20]; + } udata; + UINT8 active; //this slot is currently playing + UINT8 *base; //samples base address + UINT32 cur_addr; //current play address (24.8) + UINT32 nxt_addr; //next play address + UINT32 step; //pitch step (24.8) + UINT8 Backwards; //the wave is playing backwards + struct _EG EG; //Envelope + struct _LFO PLFO; //Phase LFO + struct _LFO ALFO; //Amplitude LFO + int slot; + signed short Prev; //Previous sample (for interpolation) +}; + + +#define MEM4B(scsp) ((scsp->udata.data[0]>>0x0)&0x0200) +#define DAC18B(scsp) ((scsp->udata.data[0]>>0x0)&0x0100) +#define MVOL(scsp) ((scsp->udata.data[0]>>0x0)&0x000F) +#define RBL(scsp) ((scsp->udata.data[1]>>0x7)&0x0003) +#define RBP(scsp) ((scsp->udata.data[1]>>0x0)&0x003F) +#define MOFULL(scsp) ((scsp->udata.data[2]>>0x0)&0x1000) +#define MOEMPTY(scsp) ((scsp->udata.data[2]>>0x0)&0x0800) +#define MIOVF(scsp) ((scsp->udata.data[2]>>0x0)&0x0400) +#define MIFULL(scsp) ((scsp->udata.data[2]>>0x0)&0x0200) +#define MIEMPTY(scsp) ((scsp->udata.data[2]>>0x0)&0x0100) + +#define SCILV0(scsp) ((scsp->udata.data[0x24/2]>>0x0)&0xff) +#define SCILV1(scsp) ((scsp->udata.data[0x26/2]>>0x0)&0xff) +#define SCILV2(scsp) ((scsp->udata.data[0x28/2]>>0x0)&0xff) + +#define SCIEX0 0 +#define SCIEX1 1 +#define SCIEX2 2 +#define SCIMID 3 +#define SCIDMA 4 +#define SCIIRQ 5 +#define SCITMA 6 +#define SCITMB 7 + +#define USEDSP + +struct _SCSP +{ + union + { + UINT16 data[0x30/2]; + UINT8 datab[0x30]; + } udata; + struct _SLOT Slots[32]; + signed short RINGBUF[64]; + unsigned char BUFPTR; +#if FM_DELAY + signed short DELAYBUF[FM_DELAY]; + unsigned char DELAYPTR; +#endif + unsigned char *SCSPRAM; + UINT32 SCSPRAM_LENGTH; + char Master; + void (*Int68kCB)(int irq); + + INT32 *buffertmpl, *buffertmpr; + + UINT32 IrqTimA; + UINT32 IrqTimBC; + UINT32 IrqMidi; + + UINT8 MidiOutW,MidiOutR; + UINT8 MidiStack[16]; + UINT8 MidiW,MidiR; + + int LPANTABLE[0x10000]; + int RPANTABLE[0x10000]; + + int TimPris[3]; + int TimCnt[3]; + + // DMA stuff + UINT32 scsp_dmea; + UINT16 scsp_drga; + UINT16 scsp_dtlg; + + int ARTABLE[64], DRTABLE[64]; + + struct _SCSPDSP DSP; +}; + +static struct _SCSP *AllocedSCSP; + +static void dma_scsp(struct _SCSP *SCSP); /*SCSP DMA transfer function*/ +#define scsp_dgate scsp_regs[0x16/2] & 0x4000 +#define scsp_ddir scsp_regs[0x16/2] & 0x2000 +#define scsp_dexe scsp_regs[0x16/2] & 0x1000 +#define dma_transfer_end ((scsp_regs[0x24/2] & 0x10)>>4)|(((scsp_regs[0x26/2] & 0x10)>>4)<<1)|(((scsp_regs[0x28/2] & 0x10)>>4)<<2) + +static const float SDLT[8]={-1000000.0,-36.0,-30.0,-24.0,-18.0,-12.0,-6.0,0.0}; + +static INT16 *bufferl; +static INT16 *bufferr; + +static int length; + +static signed short *RBUFDST; //this points to where the sample will be stored in the RingBuf + +static unsigned char DecodeSCI(struct _SCSP *SCSP,unsigned char irq) +{ + unsigned char SCI=0; + unsigned char v; + v=(SCILV0((SCSP))&(1<<irq))?1:0; + SCI|=v; + v=(SCILV1((SCSP))&(1<<irq))?1:0; + SCI|=v<<1; + v=(SCILV2((SCSP))&(1<<irq))?1:0; + SCI|=v<<2; + return SCI; +} + +static void ResetInterrupts(struct _SCSP *SCSP) +{ + UINT32 reset = SCSP->udata.data[0x22/2]; + if (reset & 0x40) + SCSP->Int68kCB(-SCSP->IrqTimA); + if (reset & 0x180) + SCSP->Int68kCB(-SCSP->IrqTimBC); +} + +static void CheckPendingIRQ(struct _SCSP *SCSP) +{ + UINT32 pend=SCSP->udata.data[0x20/2]; + UINT32 en=SCSP->udata.data[0x1e/2]; + if(SCSP->MidiW!=SCSP->MidiR) + { + SCSP->Int68kCB(SCSP->IrqMidi); + return; + } + if(!pend) + return; + if(pend&0x40) + if(en&0x40) + { + SCSP->Int68kCB(SCSP->IrqTimA); + return; + } + if(pend&0x80) + if(en&0x80) + { + SCSP->Int68kCB(SCSP->IrqTimBC); + return; + } + if(pend&0x100) + if(en&0x100) + { + SCSP->Int68kCB(SCSP->IrqTimBC); + return; + } + + SCSP->Int68kCB(0); +} + +static int Get_AR(struct _SCSP *SCSP,int base,int R) +{ + int Rate=base+(R<<1); + if(Rate>63) Rate=63; + if(Rate<0) Rate=0; + return SCSP->ARTABLE[Rate]; +} + +static int Get_DR(struct _SCSP *SCSP,int base,int R) +{ + int Rate=base+(R<<1); + if(Rate>63) Rate=63; + if(Rate<0) Rate=0; + return SCSP->DRTABLE[Rate]; +} + +static int Get_RR(struct _SCSP *SCSP,int base,int R) +{ + int Rate=base+(R<<1); + if(Rate>63) Rate=63; + if(Rate<0) Rate=0; + return SCSP->DRTABLE[Rate]; +} + +static void Compute_EG(struct _SCSP *SCSP,struct _SLOT *slot) +{ + int octave=OCT(slot); + int rate; + if(octave&8) octave=octave-16; + if(KRS(slot)!=0xf) + rate=octave+2*KRS(slot)+((FNS(slot)>>9)&1); + else + rate=0; //rate=((FNS(slot)>>9)&1); + + slot->EG.volume=0x17F<<EG_SHIFT; + slot->EG.AR=Get_AR(SCSP,rate,AR(slot)); + slot->EG.D1R=Get_DR(SCSP,rate,D1R(slot)); + slot->EG.D2R=Get_DR(SCSP,rate,D2R(slot)); + slot->EG.RR=Get_RR(SCSP,rate,RR(slot)); + slot->EG.DL=0x1f-DL(slot); + slot->EG.EGHOLD=EGHOLD(slot); +} + +static void SCSP_StopSlot(struct _SLOT *slot,int keyoff); + +static int EG_Update(struct _SLOT *slot) +{ + switch(slot->EG.state) + { + case ATTACK: + slot->EG.volume+=slot->EG.AR; + if(slot->EG.volume>=(0x3ff<<EG_SHIFT)) + { + if (!LPSLNK(slot)) + { + slot->EG.state=DECAY1; + if(slot->EG.D1R>=(1024<<EG_SHIFT)) //Skip DECAY1, go directly to DECAY2 + slot->EG.state=DECAY2; + } + slot->EG.volume=0x3ff<<EG_SHIFT; + } + if(slot->EG.EGHOLD) + return 0x3ff<<(SHIFT-10); + break; + case DECAY1: + slot->EG.volume-=slot->EG.D1R; + if(slot->EG.volume<=0) + slot->EG.volume=0; + if(slot->EG.volume>>(EG_SHIFT+5)<=slot->EG.DL) + slot->EG.state=DECAY2; + break; + case DECAY2: + if(D2R(slot)==0) + return (slot->EG.volume>>EG_SHIFT)<<(SHIFT-10); + slot->EG.volume-=slot->EG.D2R; + if(slot->EG.volume<=0) + slot->EG.volume=0; + + break; + case RELEASE: + slot->EG.volume-=slot->EG.RR; + if(slot->EG.volume<=0) + { + slot->EG.volume=0; + SCSP_StopSlot(slot,0); + //slot->EG.volume=0x17F<<EG_SHIFT; + //slot->EG.state=ATTACK; + } + break; + default: + return 1<<SHIFT; + } + return (slot->EG.volume>>EG_SHIFT)<<(SHIFT-10); +} + +static UINT32 SCSP_Step(struct _SLOT *slot) +{ + int octave=OCT(slot); + UINT64 Fn; + + Fn=(FNS_Table[FNS(slot)]); //24.8 + if(octave&8) + Fn>>=(16-octave); + else + Fn<<=octave; + + return Fn/(44100); +} + + +static void Compute_LFO(struct _SLOT *slot) +{ + if(PLFOS(slot)!=0) + LFO_ComputeStep(&(slot->PLFO),LFOF(slot),PLFOWS(slot),PLFOS(slot),0); + if(ALFOS(slot)!=0) + LFO_ComputeStep(&(slot->ALFO),LFOF(slot),ALFOWS(slot),ALFOS(slot),1); +} + +static void SCSP_StartSlot(struct _SCSP *SCSP, struct _SLOT *slot) +{ + UINT32 start_offset; + slot->active=1; + slot->Backwards=0; + slot->cur_addr=0; slot->nxt_addr=1<<SHIFT; + start_offset = PCM8B(slot) ? SA(slot) : SA(slot) & 0x7FFFE; + slot->base=SCSP->SCSPRAM + start_offset; + slot->step=SCSP_Step(slot); + Compute_EG(SCSP,slot); + slot->EG.state=ATTACK; + slot->EG.volume=0x17F<<EG_SHIFT; + slot->Prev=0; + Compute_LFO(slot); + +// printf("StartSlot: SA %x PCM8B %x LPCTL %x ALFOS %x STWINH %x TL %x EFSDL %x\n", SA(slot), PCM8B(slot), LPCTL(slot), ALFOS(slot), STWINH(slot), TL(slot), EFSDL(slot)); +// printf(" AR %x D1R %x D2R %x RR %x DL %x KRS %x EGHOLD %x LPSLNK %x\n", AR(slot), D1R(slot), D2R(slot), RR(slot), DL(slot), KRS(slot), EGHOLD(slot), LPSLNK(slot)); +} + +static void SCSP_StopSlot(struct _SLOT *slot,int keyoff) +{ + if(keyoff /*&& slot->EG.state!=RELEASE*/) + { + slot->EG.state=RELEASE; + } + else + { + slot->active=0; + } + slot->udata.data[0]&=~0x800; +} + +#define log_base_2(n) (log((float) n)/log((float) 2)) + +static void SCSP_Init(struct _SCSP *SCSP, const struct SCSPinterface *intf) +{ + int i=0; + + SCSP->IrqTimA = SCSP->IrqTimBC = SCSP->IrqMidi = 0; + SCSP->MidiR=SCSP->MidiW=0; + SCSP->MidiOutR=SCSP->MidiOutW=0; + + // get SCSP RAM + { + memset(SCSP,0,sizeof(*SCSP)); + + if (!i) + { + SCSP->Master=1; + } + else + { + SCSP->Master=0; + } + + if (intf->region) + { + SCSP->SCSPRAM = &sat_ram[0]; //(unsigned char *)intf->region; + SCSP->SCSPRAM_LENGTH = 512*1024; + SCSP->DSP.SCSPRAM = (UINT16 *)SCSP->SCSPRAM; + SCSP->DSP.SCSPRAM_LENGTH = (512*1024)/2; +// SCSP->SCSPRAM += intf->roffset; + } + } + + for(i=0;i<0x400;++i) + { + float fcent=(double) 1200.0*log_base_2((double)(((double) 1024.0+(double)i)/(double)1024.0)); + fcent=(double) 44100.0*pow(2.0,fcent/1200.0); + FNS_Table[i]=(float) (1<<SHIFT) *fcent; + } + + for(i=0;i<0x400;++i) + { + float envDB=((float)(3*(i-0x3ff)))/32.0; + float scale=(float)(1<<SHIFT); + EG_TABLE[i]=(INT32)(pow(10.0,envDB/20.0)*scale); + } + + for(i=0;i<0x10000;++i) + { + int iTL =(i>>0x0)&0xff; + int iPAN=(i>>0x8)&0x1f; + int iSDL=(i>>0xD)&0x07; + float TL=1.0; + float SegaDB=0; + float fSDL=1.0; + float PAN=1.0; + float LPAN,RPAN; + + if(iTL&0x01) SegaDB-=0.4; + if(iTL&0x02) SegaDB-=0.8; + if(iTL&0x04) SegaDB-=1.5; + if(iTL&0x08) SegaDB-=3; + if(iTL&0x10) SegaDB-=6; + if(iTL&0x20) SegaDB-=12; + if(iTL&0x40) SegaDB-=24; + if(iTL&0x80) SegaDB-=48; + + TL=pow(10.0,SegaDB/20.0); + + SegaDB=0; + if(iPAN&0x1) SegaDB-=3; + if(iPAN&0x2) SegaDB-=6; + if(iPAN&0x4) SegaDB-=12; + if(iPAN&0x8) SegaDB-=24; + + if((iPAN&0xf)==0xf) PAN=0.0; + else PAN=pow(10.0,SegaDB/20.0); + + if(iPAN<0x10) + { + LPAN=PAN; + RPAN=1.0; + } + else + { + RPAN=PAN; + LPAN=1.0; + } + + if(iSDL) + fSDL=pow(10.0,(SDLT[iSDL])/20.0); + else + fSDL=0.0; + + SCSP->LPANTABLE[i]=FIX((4.0*LPAN*TL*fSDL)); + SCSP->RPANTABLE[i]=FIX((4.0*RPAN*TL*fSDL)); + } + + SCSP->ARTABLE[0]=SCSP->DRTABLE[0]=0; //Infinite time + SCSP->ARTABLE[1]=SCSP->DRTABLE[1]=0; //Infinite time + for(i=2;i<64;++i) + { + double t,step,scale; + t=ARTimes[i]; //In ms + if(t!=0.0) + { + step=(1023*1000.0)/((float) 44100.0f*t); + scale=(double) (1<<EG_SHIFT); + SCSP->ARTABLE[i]=(int) (step*scale); + } + else + SCSP->ARTABLE[i]=1024<<EG_SHIFT; + + t=DRTimes[i]; //In ms + step=(1023*1000.0)/((float) 44100.0f*t); + scale=(double) (1<<EG_SHIFT); + SCSP->DRTABLE[i]=(int) (step*scale); + } + + // make sure all the slots are off + for(i=0;i<32;++i) + { + SCSP->Slots[i].slot=i; + SCSP->Slots[i].active=0; + SCSP->Slots[i].base=NULL; + } + + LFO_Init(); + SCSP->buffertmpl=(signed int*) malloc(44100*sizeof(signed int)); + SCSP->buffertmpr=(signed int*) malloc(44100*sizeof(signed int)); + memset(SCSP->buffertmpl,0,44100*sizeof(signed int)); + memset(SCSP->buffertmpr,0,44100*sizeof(signed int)); + + // no "pend" + SCSP[0].udata.data[0x20/2] = 0; + //SCSP[1].udata.data[0x20/2] = 0; + SCSP->TimCnt[0] = 0xffff; + SCSP->TimCnt[1] = 0xffff; + SCSP->TimCnt[2] = 0xffff; +} + +static void SCSP_UpdateSlotReg(struct _SCSP *SCSP,int s,int r) +{ + struct _SLOT *slot=SCSP->Slots+s; + int sl; + switch(r&0x3f) + { + case 0: + case 1: + if(KEYONEX(slot)) + { + for(sl=0;sl<32;++sl) + { + struct _SLOT *s2=SCSP->Slots+sl; + { + if(KEYONB(s2) && s2->EG.state==RELEASE/*&& !s2->active*/) + { + SCSP_StartSlot(SCSP, s2); + } + if(!KEYONB(s2) /*&& s2->active*/) + { + SCSP_StopSlot(s2,1); + } + } + } + slot->udata.data[0]&=~0x1000; + } + break; + case 0x10: + case 0x11: + slot->step=SCSP_Step(slot); + break; + case 0xA: + case 0xB: + slot->EG.RR=Get_RR(SCSP,0,RR(slot)); + slot->EG.DL=0x1f-DL(slot); + break; + case 0x12: + case 0x13: + Compute_LFO(slot); + break; + } +} + +static void SCSP_UpdateReg(struct _SCSP *SCSP, int reg) +{ + switch(reg&0x3f) + { + case 0x2: + case 0x3: + { + unsigned int v=RBL(SCSP); + SCSP->DSP.RBP=RBP(SCSP); + if(v==0) + SCSP->DSP.RBL=8*1024; + else if(v==1) + SCSP->DSP.RBL=16*1024; + if(v==2) + SCSP->DSP.RBL=32*1024; + if(v==3) + SCSP->DSP.RBL=64*1024; + } + break; + case 0x6: + case 0x7: + SCSP_MidiIn(0, SCSP->udata.data[0x6/2]&0xff, 0); + break; + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + break; + case 0x18: + case 0x19: + if(SCSP->Master) + { + SCSP->TimPris[0]=1<<((SCSP->udata.data[0x18/2]>>8)&0x7); + SCSP->TimCnt[0]=(SCSP->udata.data[0x18/2]&0xff)<<8; + } + break; + case 0x1a: + case 0x1b: + if(SCSP->Master) + { + SCSP->TimPris[1]=1<<((SCSP->udata.data[0x1A/2]>>8)&0x7); + SCSP->TimCnt[1]=(SCSP->udata.data[0x1A/2]&0xff)<<8; + } + break; + case 0x1C: + case 0x1D: + if(SCSP->Master) + { + SCSP->TimPris[2]=1<<((SCSP->udata.data[0x1C/2]>>8)&0x7); + SCSP->TimCnt[2]=(SCSP->udata.data[0x1C/2]&0xff)<<8; + } + break; + case 0x22: //SCIRE + case 0x23: + + if(SCSP->Master) + { + SCSP->udata.data[0x20/2]&=~SCSP->udata.data[0x22/2]; + ResetInterrupts(SCSP); + + // behavior from real hardware: if you SCIRE a timer that's expired, + // it'll immediately pop up again. ask Sakura Taisen on the Saturn... + if (SCSP->TimCnt[0] >= 0xff00) + { + SCSP->udata.data[0x20/2] |= 0x40; + } + if (SCSP->TimCnt[1] >= 0xff00) + { + SCSP->udata.data[0x20/2] |= 0x80; + } + if (SCSP->TimCnt[2] >= 0xff00) + { + SCSP->udata.data[0x20/2] |= 0x100; + } + } + break; + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + if(SCSP->Master) + { + SCSP->IrqTimA=DecodeSCI(SCSP,SCITMA); + SCSP->IrqTimBC=DecodeSCI(SCSP,SCITMB); + SCSP->IrqMidi=DecodeSCI(SCSP,SCIMID); + } + break; + } +} + +static void SCSP_UpdateSlotRegR(struct _SCSP *SCSP, int slot,int reg) +{ + +} + +static void SCSP_UpdateRegR(struct _SCSP *SCSP, int reg) +{ + switch(reg&0x3f) + { + case 4: + case 5: + { + unsigned short v=SCSP->udata.data[0x5/2]; + v&=0xff00; + v|=SCSP->MidiStack[SCSP->MidiR]; + SCSP->Int68kCB(0); // cancel the IRQ + if(SCSP->MidiR!=SCSP->MidiW) + { + ++SCSP->MidiR; + SCSP->MidiR&=15; + } + SCSP->udata.data[0x5/2]=v; + } + break; + case 8: + case 9: + { + unsigned char slot=SCSP->udata.data[0x8/2]>>11; + unsigned int CA=SCSP->Slots[slot&0x1f].cur_addr>>(SHIFT+12); + SCSP->udata.data[0x8/2]&=~(0x780); + SCSP->udata.data[0x8/2]|=CA<<7; + } + break; + } +} + +static void SCSP_w16(struct _SCSP *SCSP,unsigned int addr,unsigned short val) +{ + addr&=0xffff; + if(addr<0x400) + { + int slot=addr/0x20; + addr&=0x1f; + *((unsigned short *) (SCSP->Slots[slot].udata.datab+(addr))) = val; + SCSP_UpdateSlotReg(SCSP,slot,addr&0x1f); + } + else if(addr<0x600) + { + if (addr < 0x430) + { + *((unsigned short *) (SCSP->udata.datab+((addr&0x3f)))) = val; + SCSP_UpdateReg(SCSP, addr&0x3f); + } + } + else if(addr<0x700) + SCSP->RINGBUF[(addr-0x600)/2]=val; + else + { + //DSP + if(addr<0x780) //COEF + *((unsigned short *) (SCSP->DSP.COEF+(addr-0x700)/2))=val; + else if(addr<0x800) + *((unsigned short *) (SCSP->DSP.MADRS+(addr-0x780)/2))=val; + else if(addr<0xC00) + *((unsigned short *) (SCSP->DSP.MPRO+(addr-0x800)/2))=val; + + if(addr==0xBF0) + { +// printf("DSP start\n"); + SCSPDSP_Start(&SCSP->DSP); + } + } +} + +static unsigned short SCSP_r16(struct _SCSP *SCSP, unsigned int addr) +{ + unsigned short v=0; + addr&=0xffff; + if(addr<0x400) + { + int slot=addr/0x20; + addr&=0x1f; + SCSP_UpdateSlotRegR(SCSP, slot,addr&0x1f); + v=*((unsigned short *) (SCSP->Slots[slot].udata.datab+(addr))); + } + else if(addr<0x600) + { + if (addr < 0x430) + { + SCSP_UpdateRegR(SCSP, addr&0x3f); + v= *((unsigned short *) (SCSP->udata.datab+((addr&0x3f)))); + } + } + else if(addr<0x700) + v=SCSP->RINGBUF[(addr-0x600)/2]; + return v; +} + + +#define REVSIGN(v) ((~v)+1) + +void SCSP_TimersAddTicks(struct _SCSP *SCSP, int ticks) +{ + if(SCSP->TimCnt[0]<=0xff00) + { + SCSP->TimCnt[0] += ticks << (8-((SCSP->udata.data[0x18/2]>>8)&0x7)); + if (SCSP->TimCnt[0] > 0xFF00) + { + SCSP->TimCnt[0] = 0xFFFF; + SCSP->udata.data[0x20/2]|=0x40; + } + SCSP->udata.data[0x18/2]&=0xff00; + SCSP->udata.data[0x18/2]|=SCSP->TimCnt[0]>>8; + } + + if(SCSP->TimCnt[1]<=0xff00) + { + SCSP->TimCnt[1] += ticks << (8-((SCSP->udata.data[0x1a/2]>>8)&0x7)); + if (SCSP->TimCnt[1] > 0xFF00) + { + SCSP->TimCnt[1] = 0xFFFF; + SCSP->udata.data[0x20/2]|=0x80; + } + SCSP->udata.data[0x1a/2]&=0xff00; + SCSP->udata.data[0x1a/2]|=SCSP->TimCnt[1]>>8; + } + + if(SCSP->TimCnt[2]<=0xff00) + { + SCSP->TimCnt[2] += ticks << (8-((SCSP->udata.data[0x1c/2]>>8)&0x7)); + if (SCSP->TimCnt[2] > 0xFF00) + { + SCSP->TimCnt[2] = 0xFFFF; + SCSP->udata.data[0x20/2]|=0x100; + } + SCSP->udata.data[0x1c/2]&=0xff00; + SCSP->udata.data[0x1c/2]|=SCSP->TimCnt[2]>>8; + } +} + +INLINE INT32 SCSP_UpdateSlot(struct _SCSP *SCSP, struct _SLOT *slot) +{ + INT32 sample; + int step=slot->step; + UINT32 addr1,addr2,addr_select; // current and next sample addresses + UINT32 *addr[2] = {&addr1, &addr2}; // used for linear interpolation + UINT32 *slot_addr[2] = {&(slot->cur_addr), &(slot->nxt_addr)}; // + + if(SSCTL(slot)!=0) //no FM or noise yet + return 0; + + if(PLFOS(slot)!=0) + { + step=step*PLFO_Step(&(slot->PLFO)); + step>>=SHIFT; + } + + if(PCM8B(slot)) + { + addr1=slot->cur_addr>>SHIFT; + addr2=slot->nxt_addr>>SHIFT; + } + else + { + addr1=(slot->cur_addr>>(SHIFT-1))&0x7fffe; + addr2=(slot->nxt_addr>>(SHIFT-1))&0x7fffe; + } + + if(MDL(slot)!=0 || MDXSL(slot)!=0 || MDYSL(slot)!=0) + { + INT32 smp=(SCSP->RINGBUF[(SCSP->BUFPTR+MDXSL(slot))&63]+SCSP->RINGBUF[(SCSP->BUFPTR+MDYSL(slot))&63])/2; + + smp<<=0xA; // associate cycle with 1024 + smp>>=0x1A-MDL(slot); // ex. for MDL=0xF, sample range corresponds to +/- 64 pi (32=2^5 cycles) so shift by 11 (16-5 == 0x1A-0xF) + if(!PCM8B(slot)) smp<<=1; + + addr1+=smp; addr2+=smp; + } + + if(PCM8B(slot)) //8 bit signed + { + INT8 *p1=(signed char *) (SCSP->SCSPRAM+(((SA(slot)+addr1)^1)&0x7FFFF)); + INT8 *p2=(signed char *) (SCSP->SCSPRAM+(((SA(slot)+addr2)^1)&0x7FFFF)); + //sample=(p[0])<<8; + INT32 s; + INT32 fpart=slot->cur_addr&((1<<SHIFT)-1); + s=(int) (p1[0]<<8)*((1<<SHIFT)-fpart)+(int) (p2[0]<<8)*fpart; + sample=(s>>SHIFT); + } + else //16 bit signed (endianness?) + { + INT16 *p1=(signed short *) (SCSP->SCSPRAM+((SA(slot)+addr1)&0x7FFFE)); + INT16 *p2=(signed short *) (SCSP->SCSPRAM+((SA(slot)+addr2)&0x7FFFE)); + //sample=LE16(p[0]); + INT32 s; + INT32 fpart=slot->cur_addr&((1<<SHIFT)-1); + s=(int) LE16(p1[0])*((1<<SHIFT)-fpart)+(int) LE16(p2[0])*fpart; + sample=(s>>SHIFT); + } + + if(SBCTL(slot)&0x1) + sample ^= 0x7FFF; + if(SBCTL(slot)&0x2) + sample = (INT16)(sample^0x8000); + + if(slot->Backwards) + slot->cur_addr-=step; + else + slot->cur_addr+=step; + slot->nxt_addr=slot->cur_addr+(1<<SHIFT); + + addr1=slot->cur_addr>>SHIFT; + addr2=slot->nxt_addr>>SHIFT; + + if(addr1>=LSA(slot) && !(slot->Backwards)) + { + if(LPSLNK(slot) && slot->EG.state==ATTACK) + slot->EG.state = DECAY1; + } + + for (addr_select=0;addr_select<2;addr_select++) + { + INT32 rem_addr; + switch(LPCTL(slot)) + { + case 0: //no loop + if(*addr[addr_select]>=LSA(slot) && *addr[addr_select]>=LEA(slot)) + { + //slot->active=0; + SCSP_StopSlot(slot,0); + } + break; + case 1: //normal loop + if(*addr[addr_select]>=LEA(slot)) + { + rem_addr = *slot_addr[addr_select] - (LEA(slot)<<SHIFT); + *slot_addr[addr_select]=(LSA(slot)<<SHIFT) + rem_addr; + } + break; + case 2: //reverse loop + if((*addr[addr_select]>=LSA(slot)) && !(slot->Backwards)) + { + rem_addr = *slot_addr[addr_select] - (LSA(slot)<<SHIFT); + *slot_addr[addr_select]=(LEA(slot)<<SHIFT) - rem_addr; + slot->Backwards=1; + } + else if((*addr[addr_select]<LSA(slot) || (*slot_addr[addr_select]&0x80000000)) && slot->Backwards) + { + rem_addr = (LSA(slot)<<SHIFT) - *slot_addr[addr_select]; + *slot_addr[addr_select]=(LEA(slot)<<SHIFT) - rem_addr; + } + break; + case 3: //ping-pong + if(*addr[addr_select]>=LEA(slot)) //reached end, reverse till start + { + rem_addr = *slot_addr[addr_select] - (LEA(slot)<<SHIFT); + *slot_addr[addr_select]=(LEA(slot)<<SHIFT) - rem_addr; + slot->Backwards=1; + } + else if((*addr[addr_select]<LSA(slot) || (*slot_addr[addr_select]&0x80000000)) && slot->Backwards)//reached start or negative + { + rem_addr = (LSA(slot)<<SHIFT) - *slot_addr[addr_select]; + *slot_addr[addr_select]=(LSA(slot)<<SHIFT) + rem_addr; + slot->Backwards=0; + } + break; + } + } + + if(ALFOS(slot)!=0) + { + sample=sample*ALFO_Step(&(slot->ALFO)); + sample>>=SHIFT; + } + + if(slot->EG.state==ATTACK) + sample=(sample*EG_Update(slot))>>SHIFT; + else + sample=(sample*EG_TABLE[EG_Update(slot)>>(SHIFT-10)])>>SHIFT; + + if(!STWINH(slot)) + { + unsigned short Enc=((TL(slot))<<0x0)|(0x7<<0xd); + *RBUFDST=(sample*SCSP->LPANTABLE[Enc])>>(SHIFT+1); + } + + return sample; +} + +static void SCSP_DoMasterSamples(struct _SCSP *SCSP, int nsamples) +{ + INT16 *bufr,*bufl; + int sl, s, i; + + bufr=bufferr; + bufl=bufferl; + + for(s=0;s<nsamples;++s) + { + INT32 smpl, smpr; + + smpl = smpr = 0; + + for(sl=0;sl<32;++sl) + { +#if FM_DELAY + RBUFDST=SCSP->DELAYBUF+SCSP->DELAYPTR; +#else + RBUFDST=SCSP->RINGBUF+SCSP->BUFPTR; +#endif + if(SCSP->Slots[sl].active) + { + struct _SLOT *slot=SCSP->Slots+sl; + unsigned short Enc; + signed int sample; + + sample=SCSP_UpdateSlot(SCSP, slot); + + Enc=((TL(slot))<<0x0)|((IMXL(slot))<<0xd); + SCSPDSP_SetSample(&SCSP->DSP,(sample*SCSP->LPANTABLE[Enc])>>(SHIFT-2),ISEL(slot),IMXL(slot)); + Enc=((TL(slot))<<0x0)|((DIPAN(slot))<<0x8)|((DISDL(slot))<<0xd); + { + smpl+=(sample*SCSP->LPANTABLE[Enc])>>SHIFT; + smpr+=(sample*SCSP->RPANTABLE[Enc])>>SHIFT; + } + } + +#if FM_DELAY + SCSP->RINGBUF[(SCSP->BUFPTR+64-(FM_DELAY-1))&63] = SCSP->DELAYBUF[(SCSP->DELAYPTR+FM_DELAY-(FM_DELAY-1))%FM_DELAY]; +#endif + ++SCSP->BUFPTR; + SCSP->BUFPTR&=63; +#if FM_DELAY + ++SCSP->DELAYPTR; + if(SCSP->DELAYPTR>FM_DELAY-1) SCSP->DELAYPTR=0; +#endif + } + + SCSPDSP_Step(&SCSP->DSP); + + for(i=0;i<16;++i) + { + struct _SLOT *slot=SCSP->Slots+i; + if(EFSDL(slot)) + { + unsigned short Enc=((EFPAN(slot))<<0x8)|((EFSDL(slot))<<0xd); + smpl+=(SCSP->DSP.EFREG[i]*SCSP->LPANTABLE[Enc])>>SHIFT; + smpr+=(SCSP->DSP.EFREG[i]*SCSP->RPANTABLE[Enc])>>SHIFT; + } + } + + *bufl++ = ICLIP16(smpl>>2); + *bufr++ = ICLIP16(smpr>>2); + + SCSP_TimersAddTicks(SCSP, 1); + CheckPendingIRQ(SCSP); + } + +} + +static void dma_scsp(struct _SCSP *SCSP) +{ + static UINT16 tmp_dma[3], *scsp_regs; + + scsp_regs = (UINT16 *)SCSP->udata.datab; + + printf("SCSP: DMA transfer START\n" + "DMEA: %04x DRGA: %04x DTLG: %04x\n" + "DGATE: %d DDIR: %d\n",SCSP->scsp_dmea,SCSP->scsp_drga,SCSP->scsp_dtlg,scsp_dgate ? 1 : 0,scsp_ddir ? 1 : 0); + + /* Copy the dma values in a temp storage for resuming later * + * (DMA *can't* overwrite his parameters). */ + if(!(scsp_ddir)) + { + tmp_dma[0] = scsp_regs[0x12/2]; + tmp_dma[1] = scsp_regs[0x14/2]; + tmp_dma[2] = scsp_regs[0x16/2]; + } + + if(scsp_ddir) + { + for(;SCSP->scsp_dtlg > 0;SCSP->scsp_dtlg-=2) + { +// program_write_word(SCSP->scsp_dmea, program_read_word(0x100000|SCSP->scsp_drga)); + SCSP->scsp_dmea+=2; + SCSP->scsp_drga+=2; + } + } + else + { + for(;SCSP->scsp_dtlg > 0;SCSP->scsp_dtlg-=2) + { +// program_write_word(0x100000|SCSP->scsp_drga,program_read_word(SCSP->scsp_dmea)); + SCSP->scsp_dmea+=2; + SCSP->scsp_drga+=2; + } + } + + /*Resume the values*/ + if(!(scsp_ddir)) + { + scsp_regs[0x12/2] = tmp_dma[0]; + scsp_regs[0x14/2] = tmp_dma[1]; + scsp_regs[0x16/2] = tmp_dma[2]; + } + + /*Job done,request a dma end irq*/ +// if(scsp_regs[0x1e/2] & 0x10) +// cpunum_set_input_line(2,dma_transfer_end,HOLD_LINE); +} + +int SCSP_IRQCB(void *param) +{ + CheckPendingIRQ(param); + return -1; +} + +void SCSP_Update(void *param, INT16 **inputs, INT16 **buf, int samples) +{ + struct _SCSP *SCSP = AllocedSCSP; + bufferl = buf[0]; + bufferr = buf[1]; + length = samples; + SCSP_DoMasterSamples(SCSP, samples); +} + +void *scsp_start(const void *config) +{ + const struct SCSPinterface *intf; + + struct _SCSP *SCSP; + + SCSP = malloc(sizeof(*SCSP)); + memset(SCSP, 0, sizeof(*SCSP)); + + intf = config; + + // init the emulation + SCSP_Init(SCSP, intf); + + // set up the IRQ callbacks + { + SCSP->Int68kCB = intf->irq_callback[0]; + +// SCSP->stream = stream_create(0, 2, 44100, SCSP, SCSP_Update); + } + + AllocedSCSP = SCSP; + + return SCSP; +} + + +void SCSP_set_ram_base(int which, void *base) +{ + struct _SCSP *SCSP = AllocedSCSP; + if (SCSP) + { + SCSP->SCSPRAM = base; + SCSP->DSP.SCSPRAM = base; + } +} + + +READ16_HANDLER( SCSP_0_r ) +{ + struct _SCSP *SCSP = AllocedSCSP; + + return SCSP_r16(SCSP, offset*2); +} + +extern UINT32* stv_scu; + +WRITE16_HANDLER( SCSP_0_w ) +{ + struct _SCSP *SCSP = AllocedSCSP; + UINT16 tmp, *scsp_regs; + + tmp = SCSP_r16(SCSP, offset*2); + COMBINE_DATA(&tmp); + SCSP_w16(SCSP,offset*2, tmp); + + scsp_regs = (UINT16 *)SCSP->udata.datab; + + switch(offset*2) + { + // check DMA + case 0x412: + /*DMEA [15:1]*/ + /*Sound memory address*/ + SCSP->scsp_dmea = (((scsp_regs[0x14/2] & 0xf000)>>12)*0x10000) | (scsp_regs[0x12/2] & 0xfffe); + break; + case 0x414: + /*DMEA [19:16]*/ + SCSP->scsp_dmea = (((scsp_regs[0x14/2] & 0xf000)>>12)*0x10000) | (scsp_regs[0x12/2] & 0xfffe); + /*DRGA [11:1]*/ + /*Register memory address*/ + SCSP->scsp_drga = scsp_regs[0x14/2] & 0x0ffe; + break; + case 0x416: + /*DGATE[14]*/ + /*DDIR[13]*/ + /*if 0 sound_mem -> reg*/ + /*if 1 sound_mem <- reg*/ + /*DEXE[12]*/ + /*starting bit*/ + /*DTLG[11:1]*/ + /*size of transfer*/ + SCSP->scsp_dtlg = scsp_regs[0x16/2] & 0x0ffe; + if(scsp_dexe) + { + dma_scsp(SCSP); + scsp_regs[0x16/2]^=0x1000;//disable starting bit + } + break; + //check main cpu IRQ + case 0x42a: +#if 0 + if(stv_scu && !(stv_scu[40] & 0x40) /*&& scsp_regs[0x42c/2] & 0x20*/)/*Main CPU allow sound irq*/ + { + cpunum_set_input_line_and_vector(0, 9, HOLD_LINE , 0x46); + logerror("SCSP: Main CPU interrupt\n"); + } +#endif + break; + case 0x42c: + break; + case 0x42e: + break; + } +} + +WRITE16_HANDLER( SCSP_MidiIn ) +{ + struct _SCSP *SCSP = AllocedSCSP; + SCSP->MidiStack[SCSP->MidiW++]=data; + SCSP->MidiW &= 15; +} + +READ16_HANDLER( SCSP_MidiOutR ) +{ + struct _SCSP *SCSP = AllocedSCSP; + unsigned char val; + + val=SCSP->MidiStack[SCSP->MidiR++]; + SCSP->MidiR&=7; + return val; +} + |