/** * Furnace Tracker - multi-system chiptune tracker * Copyright (C) 2021-2023 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 _AY8930_H #define _AY8930_H #include "../dispatch.h" #include "../fixedQueue.h" #include "sound/ay8910.h" class DivPlatformAY8930: public DivDispatch { protected: struct Channel: public SharedChannel { struct Envelope { unsigned char mode; unsigned short period; short slideLow; short slide; Envelope(): mode(0), period(0), slideLow(0), slide(0) {} } envelope; struct PSGMode { // bit 3: DAC // bit 2: envelope // bit 1: noise // bit 0: tone unsigned char val; unsigned char getTone() { return (val&8)?0:(val&1); } unsigned char getNoise() { return (val&8)?0:(val&2); } unsigned char getEnvelope() { return (val&8)?0:(val&4); } PSGMode(unsigned char v=1): val(v) {} }; PSGMode curPSGMode; PSGMode nextPSGMode; struct DAC { int sample, rate, period, pos, out; bool furnaceDAC; DAC(): sample(-1), rate(0), period(0), pos(0), out(0), furnaceDAC(false) {} } dac; unsigned char autoEnvNum, autoEnvDen, duty; signed char konCycles; Channel(): SharedChannel(31), envelope(Envelope()), curPSGMode(PSGMode(0)), nextPSGMode(PSGMode(1)), dac(DAC()), autoEnvNum(0), autoEnvDen(0), duty(4), konCycles(0) {} }; Channel chan[3]; bool isMuted[3]; struct QueuedWrite { unsigned short addr; unsigned char val; bool addrOrVal; QueuedWrite(): addr(0), val(0), addrOrVal(false) {} QueuedWrite(unsigned short a, unsigned char v): addr(a), val(v), addrOrVal(false) {} }; FixedQueue writes; ay8930_device* ay; DivDispatchOscBuffer* oscBuf[3]; unsigned char regPool[32]; unsigned char ayNoiseAnd, ayNoiseOr; unsigned char stereoSep; bool bank; unsigned char sampleBank; int delay; bool extMode, stereo, clockSel; bool ioPortA, ioPortB; unsigned char portAVal, portBVal; short oldWrites[32]; short pendingWrites[32]; short* ayBuf[3]; size_t ayBufLen; void runDAC(); void checkWrites(); void updateOutSel(bool immediate=false); void immWrite(unsigned char a, unsigned char v); friend void putDispatchChip(void*,int); friend void putDispatchChan(void*,int,int); public: void acquire(short** buf, size_t len); int dispatch(DivCommand c); void* getChanState(int chan); DivDispatchOscBuffer* getOscBuffer(int chan); unsigned char* getRegisterPool(); int getRegisterPoolSize(); void reset(); void forceIns(); void tick(bool sysTick=true); void muteChannel(int ch, bool mute); void setFlags(const DivConfig& flags); int getOutputCount(); bool keyOffAffectsArp(int ch); DivMacroInt* getChanMacroInt(int ch); DivSamplePos getSamplePos(int ch); 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(); }; #endif