furnace/src/engine/platform/ay8930.h

156 lines
4.2 KiB
C
Raw Normal View History

2022-02-14 22:12:20 -05:00
/**
* Furnace Tracker - multi-system chiptune tracker
2024-01-16 21:26:57 -05:00
* Copyright (C) 2021-2024 tildearrow and contributors
2022-02-14 22:12:20 -05:00
*
* 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-14 00:02:10 -05:00
#ifndef _AY8930_H
#define _AY8930_H
#include "../dispatch.h"
2023-09-13 03:46:02 -04:00
#include "../../fixedQueue.h"
2022-01-14 00:02:10 -05:00
#include "sound/ay8910.h"
class DivPlatformAY8930: public DivDispatch {
protected:
struct Channel: public SharedChannel<int> {
struct Envelope {
unsigned char mode;
unsigned short period;
short slideLow;
short slide;
Envelope():
mode(0),
period(0),
slideLow(0),
slide(0) {}
} envelope;
struct PSGMode {
2023-08-12 00:58:44 -04:00
// bit 3: DAC
// bit 2: envelope
// bit 1: noise
// bit 0: tone
unsigned char val;
unsigned char getTone() {
2023-08-12 00:58:44 -04:00
return (val&8)?0:(val&1);
}
unsigned char getNoise() {
2023-08-12 00:58:44 -04:00
return (val&8)?0:(val&2);
}
unsigned char getEnvelope() {
2023-08-12 00:58:44 -04:00
return (val&8)?0:(val&4);
}
2023-08-12 00:58:44 -04:00
PSGMode(unsigned char v=1):
2022-09-25 18:28:04 -04:00
val(v) {}
};
2022-09-25 18:31:01 -04:00
PSGMode curPSGMode;
2022-09-25 18:28:04 -04:00
PSGMode nextPSGMode;
struct DAC {
int sample, rate, period, pos, out;
2023-08-12 00:58:44 -04:00
bool furnaceDAC;
DAC():
sample(-1),
rate(0),
period(0),
pos(0),
out(0),
2023-08-12 00:58:44 -04:00
furnaceDAC(false) {}
} dac;
unsigned char autoEnvNum, autoEnvDen, duty;
2022-01-14 00:02:10 -05:00
signed char konCycles;
Channel():
SharedChannel<int>(31),
envelope(Envelope()),
2022-09-25 18:31:01 -04:00
curPSGMode(PSGMode(0)),
2022-09-25 18:28:04 -04:00
nextPSGMode(PSGMode(1)),
dac(DAC()),
autoEnvNum(0),
autoEnvDen(0),
2022-12-17 16:42:40 -05:00
duty(4),
konCycles(0) {}
2022-01-14 00:02:10 -05:00
};
Channel chan[3];
bool isMuted[3];
struct QueuedWrite {
unsigned short addr;
unsigned char val;
bool addrOrVal;
2023-07-13 05:09:20 -04:00
QueuedWrite(): addr(0), val(0), addrOrVal(false) {}
2022-01-14 00:02:10 -05:00
QueuedWrite(unsigned short a, unsigned char v): addr(a), val(v), addrOrVal(false) {}
};
2023-07-13 05:09:20 -04:00
FixedQueue<QueuedWrite,128> writes;
2022-01-14 00:02:10 -05:00
ay8930_device* ay;
DivDispatchOscBuffer* oscBuf[3];
unsigned char regPool[32];
unsigned char ayNoiseAnd, ayNoiseOr;
2022-09-22 01:18:40 -04:00
unsigned char stereoSep;
2022-01-14 00:02:10 -05:00
bool bank;
unsigned char sampleBank;
2022-01-14 00:02:10 -05:00
int delay;
bool extMode, stereo, clockSel;
2022-03-26 21:55:43 -04:00
bool ioPortA, ioPortB;
unsigned char portAVal, portBVal;
2022-01-14 00:02:10 -05:00
short oldWrites[32];
short pendingWrites[32];
short* ayBuf[3];
size_t ayBufLen;
2022-01-27 00:29:16 -05:00
2022-09-23 21:15:20 -04:00
void runDAC();
void checkWrites();
2022-03-26 21:55:43 -04:00
void updateOutSel(bool immediate=false);
void immWrite(unsigned char a, unsigned char v);
friend void putDispatchChip(void*,int);
2022-01-27 00:29:16 -05:00
friend void putDispatchChan(void*,int,int);
2022-01-14 00:02:10 -05:00
public:
2023-01-02 04:53:37 -05:00
void acquire(short** buf, size_t len);
2022-01-14 00:02:10 -05:00
int dispatch(DivCommand c);
2022-01-27 00:29:16 -05:00
void* getChanState(int chan);
DivDispatchOscBuffer* getOscBuffer(int chan);
int mapVelocity(int ch, float vel);
unsigned char* getRegisterPool();
int getRegisterPoolSize();
2022-01-14 00:02:10 -05:00
void reset();
void forceIns();
void tick(bool sysTick=true);
2022-01-14 00:02:10 -05:00
void muteChannel(int ch, bool mute);
void setFlags(const DivConfig& flags);
int getOutputCount();
2022-01-14 00:02:10 -05:00
bool keyOffAffectsArp(int ch);
DivMacroInt* getChanMacroInt(int ch);
DivSamplePos getSamplePos(int ch);
bool getLegacyAlwaysSetVolume();
2022-01-14 00:02:10 -05:00
void notifyInsDeletion(void* ins);
2022-02-01 18:08:19 -05:00
void poke(unsigned int addr, unsigned short val);
void poke(std::vector<DivRegWrite>& wlist);
2022-02-03 18:38:57 -05:00
const char** getRegisterSheet();
int init(DivEngine* parent, int channels, int sugRate, const DivConfig& flags);
2022-01-14 00:02:10 -05:00
void quit();
};
#endif