/** * 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. */ #ifndef _AY_H #define _AY_H #include "../dispatch.h" #include "../macroInt.h" #include #include "sound/ay8910.h" class DivPlatformAY8910: public DivDispatch { protected: const unsigned char AY8914RegRemap[16]={ 0,4,1,5,2,6,9,8,11,12,13,3,7,10,14,15 }; inline unsigned char regRemap(unsigned char reg) { return intellivision?AY8914RegRemap[reg&0x0f]:reg&0x0f; } struct Channel { struct PSGMode { union { struct { unsigned char tone: 1; unsigned char noise: 1; unsigned char envelope: 1; unsigned char dac: 1; }; unsigned char val=1; }; unsigned char getTone() { return dac?0:(tone<<0); } unsigned char getNoise() { return dac?0:(noise<<1); } unsigned char getEnvelope() { return dac?0:(envelope<<2); } PSGMode(unsigned char v=0): val(v) {} }; PSGMode curPSGMode; PSGMode nextPSGMode; struct DAC { int sample, rate, period, pos, out; unsigned char furnaceDAC: 1; DAC(): sample(-1), rate(0), period(0), pos(0), out(0), furnaceDAC(0) {} } dac; int freq, baseFreq, note, pitch, pitch2; int ins; unsigned char autoEnvNum, autoEnvDen; signed char konCycles; bool active, insChanged, freqChanged, keyOn, keyOff, portaPause, inPorta; int vol, outVol; DivMacroInt std; void macroInit(DivInstrument* which) { std.init(which); pitch2=0; } Channel(): curPSGMode(PSGMode(0)), nextPSGMode(PSGMode(1)), dac(DAC()), freq(0), baseFreq(0), note(0), pitch(0), pitch2(0), ins(-1), autoEnvNum(0), autoEnvDen(0), active(false), insChanged(true), freqChanged(false), keyOn(false), keyOff(false), portaPause(false), inPorta(false), vol(0), outVol(15) {} }; Channel chan[3]; bool isMuted[3]; struct QueuedWrite { unsigned short addr; unsigned char val; bool addrOrVal; QueuedWrite(unsigned short a, unsigned char v): addr(a), val(v), addrOrVal(false) {} }; std::queue writes; ay8910_device* ay; DivDispatchOscBuffer* oscBuf[3]; unsigned char regPool[16]; unsigned char lastBusy; unsigned char sampleBank; unsigned char stereoSep; int delay; bool extMode; unsigned int extClock; unsigned char extDiv; bool stereo, sunsoft, intellivision, clockSel; bool ioPortA, ioPortB; unsigned char portAVal, portBVal; short oldWrites[16]; short pendingWrites[16]; unsigned char ayEnvMode; unsigned short ayEnvPeriod; short ayEnvSlideLow; short ayEnvSlide; short* ayBuf[3]; size_t ayBufLen; void runDAC(); void checkWrites(); void updateOutSel(bool immediate=false); friend void putDispatchChip(void*,int); friend void putDispatchChan(void*,int,int); public: void setExtClockDiv(unsigned int eclk=COLOR_NTSC, unsigned char ediv=8); void acquire(short* bufL, short* bufR, size_t start, size_t len); int dispatch(DivCommand c); void* getChanState(int chan); DivDispatchOscBuffer* getOscBuffer(int chan); unsigned char* getRegisterPool(); int getRegisterPoolSize(); void flushWrites(); void reset(); void forceIns(); void tick(bool sysTick=true); void muteChannel(int ch, bool mute); void setFlags(const DivConfig& flags); bool isStereo(); bool keyOffAffectsArp(int ch); DivMacroInt* getChanMacroInt(int ch); bool getDCOffRequired(); void notifyInsDeletion(void* ins); void poke(unsigned int addr, unsigned short val); void poke(std::vector& wlist); const char** getRegisterSheet(); int init(DivEngine* parent, int channels, int sugRate, const DivConfig& flags); void quit(); DivPlatformAY8910(bool useExtMode=false, unsigned int eclk=COLOR_NTSC, unsigned char ediv=8): DivDispatch(), extMode(useExtMode), extClock(eclk), extDiv(ediv) {} }; #endif