furnace/src/engine/dispatchContainer.cpp

354 lines
10 KiB
C++
Raw Normal View History

2022-02-15 03:12:20 +00:00
/**
* Furnace Tracker - multi-system chiptune tracker
* Copyright (C) 2021-2022 tildearrow and contributors
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
2022-01-17 06:42:26 +00:00
#include "blip_buf.h"
2022-01-08 21:03:32 +00:00
#include "engine.h"
#include "platform/genesis.h"
#include "platform/genesisext.h"
#include "platform/sms.h"
#include "platform/opll.h"
2022-01-08 21:03:32 +00:00
#include "platform/gb.h"
#include "platform/pce.h"
#include "platform/nes.h"
#include "platform/c64.h"
#include "platform/arcade.h"
#include "platform/tx81z.h"
2022-01-08 21:03:32 +00:00
#include "platform/ym2610.h"
#include "platform/ym2610ext.h"
#include "platform/ym2610b.h"
#include "platform/ym2610bext.h"
#include "platform/ay.h"
2022-01-14 05:02:10 +00:00
#include "platform/ay8930.h"
#include "platform/opl.h"
#include "platform/tia.h"
2022-01-14 21:29:27 +00:00
#include "platform/saa.h"
2022-01-15 22:28:33 +00:00
#include "platform/amiga.h"
2022-03-04 23:18:43 +00:00
#include "platform/pcspkr.h"
#include "platform/segapcm.h"
2022-02-22 09:01:57 +00:00
#include "platform/qsound.h"
2022-03-04 11:13:49 +00:00
#include "platform/vera.h"
#include "platform/x1_010.h"
#include "platform/su.h"
2022-03-07 12:04:20 +00:00
#include "platform/swan.h"
2022-02-20 17:15:15 +00:00
#include "platform/lynx.h"
#include "platform/bubsyswsg.h"
2022-03-22 16:48:45 +00:00
#include "platform/n163.h"
2022-03-21 14:02:51 +00:00
#include "platform/pet.h"
2022-03-22 07:48:48 +00:00
#include "platform/vic20.h"
#include "platform/vrc6.h"
2022-04-04 03:37:16 +00:00
#include "platform/fds.h"
2022-04-06 05:34:12 +00:00
#include "platform/mmc5.h"
2022-05-10 04:18:25 +00:00
#include "platform/scc.h"
2022-01-08 21:03:32 +00:00
#include "platform/dummy.h"
#include "../ta-log.h"
#include "song.h"
2022-01-08 21:03:32 +00:00
void DivDispatchContainer::setRates(double gotRate) {
blip_set_rates(bb[0],dispatch->rate,gotRate);
blip_set_rates(bb[1],dispatch->rate,gotRate);
}
2022-01-17 06:42:26 +00:00
void DivDispatchContainer::setQuality(bool lowQual) {
lowQuality=lowQual;
}
void DivDispatchContainer::acquire(size_t offset, size_t count) {
dispatch->acquire(bbIn[0],bbIn[1],offset,count);
}
void DivDispatchContainer::flush(size_t count) {
blip_read_samples(bb[0],bbOut[0],count,0);
if (dispatch->isStereo()) {
blip_read_samples(bb[1],bbOut[1],count,0);
}
}
void DivDispatchContainer::fillBuf(size_t runtotal, size_t offset, size_t size) {
2022-04-09 23:25:25 +00:00
if (dcOffCompensation && runtotal>0) {
dcOffCompensation=false;
prevSample[0]=bbIn[0][0];
if (dispatch->isStereo()) prevSample[1]=bbIn[1][0];
}
2022-01-17 06:42:26 +00:00
if (lowQuality) {
for (size_t i=0; i<runtotal; i++) {
temp[0]=bbIn[0][i];
blip_add_delta_fast(bb[0],i,temp[0]-prevSample[0]);
prevSample[0]=temp[0];
}
if (dispatch->isStereo()) for (size_t i=0; i<runtotal; i++) {
temp[1]=bbIn[1][i];
blip_add_delta_fast(bb[1],i,temp[1]-prevSample[1]);
prevSample[1]=temp[1];
}
} else {
for (size_t i=0; i<runtotal; i++) {
temp[0]=bbIn[0][i];
blip_add_delta(bb[0],i,temp[0]-prevSample[0]);
prevSample[0]=temp[0];
}
2022-01-17 06:42:26 +00:00
if (dispatch->isStereo()) for (size_t i=0; i<runtotal; i++) {
temp[1]=bbIn[1][i];
blip_add_delta(bb[1],i,temp[1]-prevSample[1]);
prevSample[1]=temp[1];
}
}
blip_end_frame(bb[0],runtotal);
blip_read_samples(bb[0],bbOut[0]+offset,size,0);
/*if (totalRead<(int)size && totalRead>0) {
for (size_t i=totalRead; i<size; i++) {
bbOut[0][i]=bbOut[0][totalRead-1];//bbOut[0][totalRead];
}
}*/
if (dispatch->isStereo()) {
blip_end_frame(bb[1],runtotal);
blip_read_samples(bb[1],bbOut[1]+offset,size,0);
}
}
2022-01-08 21:03:32 +00:00
void DivDispatchContainer::clear() {
blip_clear(bb[0]);
blip_clear(bb[1]);
temp[0]=0;
temp[1]=0;
prevSample[0]=0;
prevSample[1]=0;
2022-04-09 23:25:25 +00:00
if (dispatch->getDCOffRequired()) {
dcOffCompensation=true;
}
2022-01-20 02:04:51 +00:00
// run for one cycle to determine DC offset
// TODO: SAA1099 doesn't like that
/*dispatch->acquire(bbIn[0],bbIn[1],0,1);
2022-01-20 02:04:51 +00:00
temp[0]=bbIn[0][0];
temp[1]=bbIn[1][0];
prevSample[0]=temp[0];
prevSample[1]=temp[1];*/
2022-01-08 21:03:32 +00:00
}
void DivDispatchContainer::init(DivSystem sys, DivEngine* eng, int chanCount, double gotRate, unsigned int flags) {
2022-01-08 21:03:32 +00:00
if (dispatch!=NULL) return;
bb[0]=blip_new(32768);
if (bb[0]==NULL) {
logE("not enough memory!");
2022-01-08 21:03:32 +00:00
return;
}
bb[1]=blip_new(32768);
if (bb[1]==NULL) {
logE("not enough memory!");
2022-01-08 21:03:32 +00:00
return;
}
bbOut[0]=new short[32768];
bbOut[1]=new short[32768];
bbIn[0]=new short[32768];
bbIn[1]=new short[32768];
bbInLen=32768;
switch (sys) {
case DIV_SYSTEM_YMU759:
dispatch=new DivPlatformOPL;
((DivPlatformOPL*)dispatch)->setOPLType(759,false);
break;
case DIV_SYSTEM_YM2612:
2022-01-08 21:03:32 +00:00
dispatch=new DivPlatformGenesis;
((DivPlatformGenesis*)dispatch)->setYMFM(eng->getConfInt("ym2612Core",0));
2022-01-08 21:03:32 +00:00
break;
case DIV_SYSTEM_YM2612_EXT:
2022-01-08 21:03:32 +00:00
dispatch=new DivPlatformGenesisExt;
((DivPlatformGenesisExt*)dispatch)->setYMFM(eng->getConfInt("ym2612Core",0));
2022-01-08 21:03:32 +00:00
break;
case DIV_SYSTEM_SMS:
dispatch=new DivPlatformSMS;
break;
case DIV_SYSTEM_GB:
dispatch=new DivPlatformGB;
break;
case DIV_SYSTEM_PCE:
dispatch=new DivPlatformPCE;
break;
case DIV_SYSTEM_NES:
dispatch=new DivPlatformNES;
((DivPlatformNES*)dispatch)->setNSFPlay(eng->getConfInt("nesCore",0)==1);
2022-01-08 21:03:32 +00:00
break;
case DIV_SYSTEM_C64_6581:
dispatch=new DivPlatformC64;
((DivPlatformC64*)dispatch)->setChipModel(true);
break;
case DIV_SYSTEM_C64_8580:
dispatch=new DivPlatformC64;
((DivPlatformC64*)dispatch)->setChipModel(false);
break;
case DIV_SYSTEM_YM2151:
2022-01-08 21:03:32 +00:00
dispatch=new DivPlatformArcade;
2022-01-17 21:27:57 +00:00
((DivPlatformArcade*)dispatch)->setYMFM(eng->getConfInt("arcadeCore",0)==0);
2022-01-08 21:03:32 +00:00
break;
case DIV_SYSTEM_YM2610:
case DIV_SYSTEM_YM2610_FULL:
2022-01-08 21:03:32 +00:00
dispatch=new DivPlatformYM2610;
break;
case DIV_SYSTEM_YM2610_EXT:
case DIV_SYSTEM_YM2610_FULL_EXT:
2022-01-08 21:03:32 +00:00
dispatch=new DivPlatformYM2610Ext;
break;
case DIV_SYSTEM_YM2610B:
dispatch=new DivPlatformYM2610B;
break;
case DIV_SYSTEM_YM2610B_EXT:
dispatch=new DivPlatformYM2610BExt;
break;
2022-01-15 22:28:33 +00:00
case DIV_SYSTEM_AMIGA:
dispatch=new DivPlatformAmiga;
break;
case DIV_SYSTEM_AY8910:
dispatch=new DivPlatformAY8910;
break;
2022-01-14 05:02:10 +00:00
case DIV_SYSTEM_AY8930:
dispatch=new DivPlatformAY8930;
break;
2022-04-04 03:37:16 +00:00
case DIV_SYSTEM_FDS:
dispatch=new DivPlatformFDS;
2022-05-02 21:53:55 +00:00
((DivPlatformFDS*)dispatch)->setNSFPlay(eng->getConfInt("fdsCore",0)==1);
2022-04-04 03:37:16 +00:00
break;
case DIV_SYSTEM_TIA:
dispatch=new DivPlatformTIA;
break;
case DIV_SYSTEM_OPLL:
case DIV_SYSTEM_OPLL_DRUMS:
case DIV_SYSTEM_VRC7:
dispatch=new DivPlatformOPLL;
2022-02-28 07:19:27 +00:00
((DivPlatformOPLL*)dispatch)->setVRC7(sys==DIV_SYSTEM_VRC7);
((DivPlatformOPLL*)dispatch)->setProperDrums(sys==DIV_SYSTEM_OPLL_DRUMS);
break;
2022-03-07 22:07:29 +00:00
case DIV_SYSTEM_OPL:
dispatch=new DivPlatformOPL;
((DivPlatformOPL*)dispatch)->setOPLType(1,false);
break;
case DIV_SYSTEM_OPL_DRUMS:
dispatch=new DivPlatformOPL;
((DivPlatformOPL*)dispatch)->setOPLType(1,true);
break;
case DIV_SYSTEM_OPL2:
dispatch=new DivPlatformOPL;
((DivPlatformOPL*)dispatch)->setOPLType(2,false);
break;
case DIV_SYSTEM_OPL2_DRUMS:
dispatch=new DivPlatformOPL;
((DivPlatformOPL*)dispatch)->setOPLType(2,true);
break;
case DIV_SYSTEM_OPL3:
dispatch=new DivPlatformOPL;
2022-03-07 22:07:29 +00:00
((DivPlatformOPL*)dispatch)->setOPLType(3,false);
break;
case DIV_SYSTEM_OPL3_DRUMS:
dispatch=new DivPlatformOPL;
((DivPlatformOPL*)dispatch)->setOPLType(3,true);
break;
case DIV_SYSTEM_OPZ:
dispatch=new DivPlatformTX81Z;
break;
case DIV_SYSTEM_SAA1099: {
int saaCore=eng->getConfInt("saaCore",1);
if (saaCore<0 || saaCore>2) saaCore=0;
2022-01-14 21:29:27 +00:00
dispatch=new DivPlatformSAA1099;
((DivPlatformSAA1099*)dispatch)->setCore((DivSAACores)saaCore);
2022-01-14 21:29:27 +00:00
break;
}
2022-03-04 23:18:43 +00:00
case DIV_SYSTEM_PCSPKR:
dispatch=new DivPlatformPCSpeaker;
break;
2022-02-20 17:15:15 +00:00
case DIV_SYSTEM_LYNX:
dispatch=new DivPlatformLynx;
break;
2022-02-22 09:01:57 +00:00
case DIV_SYSTEM_QSOUND:
dispatch=new DivPlatformQSound;
2022-02-20 17:15:15 +00:00
break;
case DIV_SYSTEM_SEGAPCM:
case DIV_SYSTEM_SEGAPCM_COMPAT:
dispatch=new DivPlatformSegaPCM;
break;
case DIV_SYSTEM_X1_010:
dispatch=new DivPlatformX1_010;
break;
2022-03-06 16:13:47 +00:00
case DIV_SYSTEM_SWAN:
2022-03-07 12:04:20 +00:00
dispatch=new DivPlatformSwan;
2022-03-06 16:13:47 +00:00
break;
2022-03-04 11:13:49 +00:00
case DIV_SYSTEM_VERA:
dispatch=new DivPlatformVERA;
2022-03-04 11:13:49 +00:00
break;
case DIV_SYSTEM_BUBSYS_WSG:
dispatch=new DivPlatformBubSysWSG;
break;
2022-03-22 16:48:45 +00:00
case DIV_SYSTEM_N163:
dispatch=new DivPlatformN163;
break;
2022-03-21 14:02:51 +00:00
case DIV_SYSTEM_PET:
dispatch=new DivPlatformPET;
break;
2022-03-22 07:48:48 +00:00
case DIV_SYSTEM_VIC20:
dispatch=new DivPlatformVIC20;
break;
case DIV_SYSTEM_VRC6:
dispatch=new DivPlatformVRC6;
break;
2022-04-06 05:34:12 +00:00
case DIV_SYSTEM_MMC5:
dispatch=new DivPlatformMMC5;
break;
2022-05-10 04:18:25 +00:00
case DIV_SYSTEM_SCC:
dispatch=new DivPlatformSCC;
((DivPlatformSCC*)dispatch)->setChipModel(false);
break;
case DIV_SYSTEM_SCC_PLUS:
dispatch=new DivPlatformSCC;
((DivPlatformSCC*)dispatch)->setChipModel(true);
break;
case DIV_SYSTEM_SOUND_UNIT:
dispatch=new DivPlatformSoundUnit;
break;
case DIV_SYSTEM_DUMMY:
dispatch=new DivPlatformDummy;
break;
2022-01-08 21:03:32 +00:00
default:
logW("this system is not supported yet! using dummy platform.");
2022-01-08 21:03:32 +00:00
dispatch=new DivPlatformDummy;
break;
}
dispatch->init(eng,chanCount,gotRate,flags);
2022-01-08 21:03:32 +00:00
}
void DivDispatchContainer::quit() {
if (dispatch==NULL) return;
dispatch->quit();
delete dispatch;
dispatch=NULL;
delete[] bbOut[0];
delete[] bbOut[1];
delete[] bbIn[0];
delete[] bbIn[1];
bbInLen=0;
blip_delete(bb[0]);
blip_delete(bb[1]);
}