2022-04-20 16:52:37 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _ES5506_H
|
|
|
|
#define _ES5506_H
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "../dispatch.h"
|
|
|
|
#include "../engine.h"
|
|
|
|
#include <queue>
|
|
|
|
#include "../macroInt.h"
|
|
|
|
#include "../sample.h"
|
|
|
|
#include "sound/es550x/es5506.hpp"
|
|
|
|
|
|
|
|
class DivPlatformES5506: public DivDispatch, public es550x_intf {
|
|
|
|
struct Channel {
|
|
|
|
struct PCM {
|
2022-05-22 09:24:17 +00:00
|
|
|
bool isNoteMap;
|
2022-04-30 14:45:05 +00:00
|
|
|
int index, next;
|
2022-05-10 15:16:20 +00:00
|
|
|
int note;
|
2022-04-26 03:04:23 +00:00
|
|
|
double freqOffs;
|
2022-05-10 15:16:20 +00:00
|
|
|
double nextFreqOffs;
|
2022-05-17 18:09:55 +00:00
|
|
|
bool reversed, pause;
|
2022-04-20 16:52:37 +00:00
|
|
|
unsigned int bank;
|
2022-04-26 03:04:23 +00:00
|
|
|
unsigned int start;
|
|
|
|
unsigned int end;
|
|
|
|
unsigned int length;
|
2022-04-20 16:52:37 +00:00
|
|
|
unsigned int loopStart;
|
|
|
|
unsigned int loopEnd;
|
|
|
|
DivSampleLoopMode loopMode;
|
|
|
|
PCM():
|
2022-05-22 09:24:17 +00:00
|
|
|
isNoteMap(false),
|
2022-04-20 16:52:37 +00:00
|
|
|
index(-1),
|
2022-04-30 14:45:05 +00:00
|
|
|
next(-1),
|
2022-05-10 15:16:20 +00:00
|
|
|
note(0),
|
2022-04-26 03:04:23 +00:00
|
|
|
freqOffs(1.0),
|
2022-05-10 15:16:20 +00:00
|
|
|
nextFreqOffs(1.0),
|
2022-04-26 03:04:23 +00:00
|
|
|
reversed(false),
|
2022-05-17 18:09:55 +00:00
|
|
|
pause(false),
|
2022-04-20 16:52:37 +00:00
|
|
|
bank(0),
|
2022-04-26 03:04:23 +00:00
|
|
|
start(0),
|
|
|
|
end(0),
|
|
|
|
length(0),
|
2022-04-20 16:52:37 +00:00
|
|
|
loopStart(0),
|
|
|
|
loopEnd(0),
|
|
|
|
loopMode(DIV_SAMPLE_LOOPMODE_ONESHOT) {}
|
|
|
|
} pcm;
|
2022-05-31 04:48:35 +00:00
|
|
|
int freq, baseFreq, nextFreq, pitch, pitch2, note, nextNote, currNote, ins, wave;
|
2022-05-03 18:32:26 +00:00
|
|
|
unsigned int volMacroMax, panMacroMax;
|
2022-05-27 04:01:06 +00:00
|
|
|
bool active, insChanged, freqChanged, keyOn, keyOff, inPorta, useWave, isReverseLoop, isTranswave, transwaveIRQ;
|
2022-05-22 09:24:17 +00:00
|
|
|
unsigned int cr;
|
2022-04-30 14:45:05 +00:00
|
|
|
|
2022-05-10 15:16:20 +00:00
|
|
|
struct NoteChanged { // Note changed flags
|
|
|
|
union { // pack flag bits in single byte
|
|
|
|
struct { // flag bits
|
|
|
|
unsigned char offs: 1; // frequency offset
|
|
|
|
unsigned char note: 1; // note
|
|
|
|
unsigned char freq: 1; // base frequency
|
|
|
|
unsigned char dummy: 5; // dummy for bit padding
|
|
|
|
};
|
|
|
|
unsigned char changed; // Packed flags are stored here
|
|
|
|
};
|
|
|
|
|
|
|
|
NoteChanged() :
|
|
|
|
changed(0) {}
|
|
|
|
} noteChanged;
|
|
|
|
|
2022-04-30 14:45:05 +00:00
|
|
|
struct VolChanged { // Volume changed flags
|
|
|
|
union { // pack flag bits in single byte
|
|
|
|
struct { // flag bits
|
|
|
|
unsigned char lVol: 1; // left volume
|
|
|
|
unsigned char rVol: 1; // right volume
|
|
|
|
unsigned char dummy: 6; // dummy for bit padding
|
|
|
|
};
|
|
|
|
unsigned char changed; // Packed flags are stored here
|
|
|
|
};
|
|
|
|
|
|
|
|
VolChanged() :
|
|
|
|
changed(0) {}
|
|
|
|
} volChanged;
|
2022-04-26 03:04:23 +00:00
|
|
|
|
|
|
|
struct FilterChanged { // Filter changed flags
|
|
|
|
union { // pack flag bits in single byte
|
|
|
|
struct { // flag bits
|
|
|
|
unsigned char mode: 1; // Filter mode
|
|
|
|
unsigned char k1: 1; // K1
|
|
|
|
unsigned char k2: 1; // K2
|
|
|
|
unsigned char dummy: 5; // dummy for bit padding
|
|
|
|
};
|
|
|
|
unsigned char changed; // Packed flags are stored here
|
|
|
|
};
|
|
|
|
|
|
|
|
FilterChanged():
|
|
|
|
changed(0) {}
|
|
|
|
} filterChanged;
|
|
|
|
|
|
|
|
struct EnvChanged { // Envelope changed flags
|
|
|
|
union { // pack flag bits in single byte
|
|
|
|
struct { // flag bits
|
|
|
|
unsigned char ecount: 1; // Envelope count
|
|
|
|
unsigned char lVRamp: 1; // Left volume Ramp
|
|
|
|
unsigned char rVRamp: 1; // Right volume Ramp
|
|
|
|
unsigned char k1Ramp: 1; // K1 Ramp w/Slow flag
|
|
|
|
unsigned char k2Ramp: 1; // K2 Ramp w/Slow flag
|
|
|
|
unsigned char dummy: 3; // dummy for bit padding
|
|
|
|
};
|
|
|
|
unsigned char changed; // Packed flags are stored here
|
|
|
|
};
|
|
|
|
|
|
|
|
EnvChanged():
|
|
|
|
changed(0) {}
|
|
|
|
} envChanged;
|
|
|
|
|
2022-05-24 15:52:00 +00:00
|
|
|
struct PCMChanged {
|
2022-05-22 09:24:17 +00:00
|
|
|
union {
|
|
|
|
struct {
|
|
|
|
unsigned char index: 1; // sample index
|
|
|
|
unsigned char slice: 1; // transwave slice
|
|
|
|
unsigned char position: 1; // sample position in memory
|
|
|
|
unsigned char loopBank: 1; // Loop mode and Bank
|
|
|
|
unsigned char transwaveInd: 1; // transwave index
|
2022-05-24 15:52:00 +00:00
|
|
|
unsigned char dummy: 3; // dummy for bit padding
|
2022-05-22 09:24:17 +00:00
|
|
|
};
|
|
|
|
unsigned char changed;
|
|
|
|
};
|
|
|
|
} pcmChanged;
|
|
|
|
|
2022-04-26 03:04:23 +00:00
|
|
|
signed int k1Offs, k2Offs;
|
2022-05-03 02:07:50 +00:00
|
|
|
signed int k1Slide, k2Slide;
|
|
|
|
signed int k1Prev, k2Prev;
|
2022-04-29 17:32:55 +00:00
|
|
|
unsigned int vol, lVol, rVol;
|
|
|
|
unsigned int outVol, outLVol, outRVol;
|
|
|
|
unsigned int resLVol, resRVol;
|
2022-05-01 12:26:10 +00:00
|
|
|
signed int lOut, rOut, oscOut;
|
2022-04-20 16:52:37 +00:00
|
|
|
DivInstrumentES5506::Filter filter;
|
|
|
|
DivInstrumentES5506::Envelope envelope;
|
2022-05-10 15:16:20 +00:00
|
|
|
DivInstrumentAmiga::TransWave transWave;
|
2022-04-20 16:52:37 +00:00
|
|
|
DivMacroInt std;
|
2022-04-28 14:30:50 +00:00
|
|
|
void macroInit(DivInstrument* which) {
|
|
|
|
std.init(which);
|
|
|
|
pitch2=0;
|
|
|
|
if (std.ex1.mode==2) {
|
|
|
|
k1Offs=0;
|
|
|
|
}
|
|
|
|
if (std.ex1.mode==2) {
|
|
|
|
k2Offs=0;
|
|
|
|
}
|
2022-05-03 02:07:50 +00:00
|
|
|
k1Prev=0xffff;
|
|
|
|
k2Prev=0xffff;
|
2022-04-28 14:30:50 +00:00
|
|
|
}
|
2022-04-20 16:52:37 +00:00
|
|
|
Channel():
|
|
|
|
freq(0),
|
|
|
|
baseFreq(0),
|
2022-05-10 15:16:20 +00:00
|
|
|
nextFreq(0),
|
2022-04-20 16:52:37 +00:00
|
|
|
pitch(0),
|
2022-04-28 14:30:50 +00:00
|
|
|
pitch2(0),
|
2022-04-20 16:52:37 +00:00
|
|
|
note(0),
|
2022-05-10 15:16:20 +00:00
|
|
|
nextNote(0),
|
2022-05-31 04:48:35 +00:00
|
|
|
currNote(0),
|
2022-04-26 03:54:11 +00:00
|
|
|
ins(-1),
|
|
|
|
wave(-1),
|
2022-05-03 18:32:26 +00:00
|
|
|
volMacroMax(0xffff),
|
|
|
|
panMacroMax(0xffff),
|
2022-04-20 16:52:37 +00:00
|
|
|
active(false),
|
|
|
|
insChanged(true),
|
|
|
|
freqChanged(false),
|
|
|
|
keyOn(false),
|
|
|
|
keyOff(false),
|
|
|
|
inPorta(false),
|
2022-04-28 14:30:50 +00:00
|
|
|
useWave(false),
|
|
|
|
isReverseLoop(false),
|
2022-05-22 09:24:17 +00:00
|
|
|
isTranswave(false),
|
|
|
|
transwaveIRQ(false),
|
|
|
|
cr(0),
|
2022-04-26 03:04:23 +00:00
|
|
|
k1Offs(0),
|
|
|
|
k2Offs(0),
|
2022-05-03 02:07:50 +00:00
|
|
|
k1Slide(0),
|
|
|
|
k2Slide(0),
|
|
|
|
k1Prev(0xffff),
|
|
|
|
k2Prev(0xffff),
|
2022-04-26 03:04:23 +00:00
|
|
|
vol(0xff),
|
2022-04-30 14:45:05 +00:00
|
|
|
lVol(0xff),
|
|
|
|
rVol(0xff),
|
2022-04-20 16:52:37 +00:00
|
|
|
outVol(0xffff),
|
|
|
|
outLVol(0xffff),
|
|
|
|
outRVol(0xffff),
|
|
|
|
resLVol(0xffff),
|
2022-05-01 12:26:10 +00:00
|
|
|
resRVol(0xffff),
|
|
|
|
lOut(0),
|
|
|
|
rOut(0),
|
|
|
|
oscOut(0) {}
|
2022-04-20 16:52:37 +00:00
|
|
|
};
|
|
|
|
Channel chan[32];
|
2022-05-01 12:26:10 +00:00
|
|
|
DivDispatchOscBuffer* oscBuf[32];
|
2022-04-20 16:52:37 +00:00
|
|
|
bool isMuted[32];
|
2022-05-03 02:07:50 +00:00
|
|
|
signed short* sampleMem; // ES5506 uses 16 bit data bus for samples
|
|
|
|
size_t sampleMemLen;
|
2022-04-20 16:52:37 +00:00
|
|
|
struct QueuedHostIntf {
|
|
|
|
unsigned char step;
|
|
|
|
unsigned char addr;
|
|
|
|
unsigned int val;
|
|
|
|
unsigned int mask;
|
|
|
|
unsigned int* read;
|
|
|
|
unsigned short delay;
|
|
|
|
bool isRead;
|
|
|
|
QueuedHostIntf(unsigned char s, unsigned char a, unsigned int v, unsigned int m=(unsigned int)(~0), unsigned short d=0):
|
|
|
|
step(s),
|
|
|
|
addr(a),
|
|
|
|
val(v),
|
|
|
|
mask(m),
|
|
|
|
read(NULL),
|
|
|
|
delay(0),
|
|
|
|
isRead(false) {}
|
|
|
|
QueuedHostIntf(unsigned char s, unsigned char a, unsigned int* r, unsigned int m=(unsigned int)(~0), unsigned short d=0):
|
|
|
|
step(s),
|
|
|
|
addr(a),
|
|
|
|
val(0),
|
|
|
|
mask(m),
|
|
|
|
read(r),
|
|
|
|
delay(d),
|
|
|
|
isRead(true) {}
|
|
|
|
};
|
|
|
|
std::queue<QueuedHostIntf> hostIntf32;
|
|
|
|
std::queue<QueuedHostIntf> hostIntf8;
|
|
|
|
int cycle, curPage;
|
|
|
|
unsigned char maskedVal;
|
|
|
|
unsigned int irqv;
|
|
|
|
bool isMasked, isReaded;
|
|
|
|
bool irqTrigger;
|
2022-05-22 09:24:17 +00:00
|
|
|
unsigned int curCR;
|
|
|
|
unsigned char transwaveCh;
|
2022-05-01 12:26:10 +00:00
|
|
|
unsigned char prevChanCycle;
|
2022-04-20 16:52:37 +00:00
|
|
|
|
|
|
|
unsigned char initChanMax, chanMax;
|
|
|
|
|
|
|
|
es5506_core es5506;
|
|
|
|
unsigned char regPool[4*16*128]; // 7 bit page x 16 registers per page x 32 bit per registers
|
|
|
|
|
|
|
|
friend void putDispatchChan(void*,int,int);
|
|
|
|
|
|
|
|
public:
|
2022-04-27 05:29:53 +00:00
|
|
|
virtual void e_pin(bool state) override; // E output
|
2022-04-20 16:52:37 +00:00
|
|
|
|
|
|
|
virtual void irqb(bool state) override; // IRQB output
|
|
|
|
virtual s16 read_sample(u8 voice, u8 bank, u32 address) override {
|
2022-05-03 02:07:50 +00:00
|
|
|
if (sampleMem==NULL) return 0;
|
|
|
|
return sampleMem[((bank&3)<<21)|(address&0x1fffff)];
|
2022-04-20 16:52:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void acquire(short* bufL, short* bufR, size_t start, size_t len) override;
|
|
|
|
virtual int dispatch(DivCommand c) override;
|
|
|
|
virtual void* getChanState(int chan) override;
|
2022-05-01 12:26:10 +00:00
|
|
|
virtual DivDispatchOscBuffer* getOscBuffer(int chan) override;
|
2022-04-20 16:52:37 +00:00
|
|
|
virtual unsigned char* getRegisterPool() override;
|
|
|
|
virtual int getRegisterPoolSize() override;
|
|
|
|
virtual void reset() override;
|
|
|
|
virtual void forceIns() override;
|
2022-04-26 03:04:23 +00:00
|
|
|
virtual void tick(bool sysTick=true) override;
|
2022-04-20 16:52:37 +00:00
|
|
|
virtual void muteChannel(int ch, bool mute) override;
|
|
|
|
virtual bool isStereo() override;
|
|
|
|
virtual bool keyOffAffectsArp(int ch) override;
|
|
|
|
virtual void setFlags(unsigned int flags) override;
|
|
|
|
virtual void notifyInsChange(int ins) override;
|
|
|
|
virtual void notifyWaveChange(int wave) override;
|
|
|
|
virtual void notifyInsDeletion(void* ins) override;
|
|
|
|
virtual void poke(unsigned int addr, unsigned short val) override;
|
|
|
|
virtual void poke(std::vector<DivRegWrite>& wlist) override;
|
2022-05-03 02:07:50 +00:00
|
|
|
virtual const void* getSampleMem(int index = 0) override;
|
|
|
|
virtual size_t getSampleMemCapacity(int index = 0) override;
|
|
|
|
virtual size_t getSampleMemUsage(int index = 0) override;
|
|
|
|
virtual void renderSamples() override;
|
2022-04-20 16:52:37 +00:00
|
|
|
virtual const char** getRegisterSheet() override;
|
|
|
|
virtual const char* getEffectName(unsigned char effect) override;
|
|
|
|
virtual int init(DivEngine* parent, int channels, int sugRate, unsigned int flags) override;
|
|
|
|
virtual void quit() override;
|
|
|
|
DivPlatformES5506():
|
|
|
|
DivDispatch(),
|
|
|
|
es550x_intf(),
|
|
|
|
es5506(*this) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|