furnace/src/engine/platform/opl.h

218 lines
6.6 KiB
C
Raw Normal View History

/**
* Furnace Tracker - multi-system chiptune tracker
2024-01-16 21:26:57 -05:00
* Copyright (C) 2021-2024 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 _OPL_H
#define _OPL_H
#include "../dispatch.h"
2023-09-13 02:46:02 -05:00
#include "../../fixedQueue.h"
#include "../../../extern/opl/opl3.h"
2023-11-22 19:28:36 -05:00
extern "C" {
#include "../../../extern/YM3812-LLE/fmopl2.h"
#include "../../../extern/YMF262-LLE/fmopl3.h"
}
2022-05-13 14:59:36 -05:00
#include "sound/ymfm/ymfm_adpcm.h"
2023-11-22 19:28:36 -05:00
#include "sound/ymfm/ymfm_opl.h"
#include "sound/ymfm/ymfm_pcm.h"
#include "sound/ymf278b/ymf278.h"
2022-05-13 14:59:36 -05:00
class DivOPLAInterface: public ymfm::ymfm_interface {
public:
unsigned char* adpcmBMem;
unsigned char* pcmMem;
2022-05-13 14:59:36 -05:00
int sampleBank;
uint8_t ymfm_external_read(ymfm::access_class type, uint32_t address);
void ymfm_external_write(ymfm::access_class type, uint32_t address, uint8_t data);
DivOPLAInterface(): adpcmBMem(NULL), pcmMem(NULL), sampleBank(0) {}
2022-05-13 14:59:36 -05:00
};
class DivYMF278MemoryInterface: public MemoryInterface {
public:
unsigned char* memory;
DivYMF278MemoryInterface(unsigned size_) : memory(NULL), size(size_) {};
byte operator[](unsigned address) const override;
unsigned getSize() const override { return size; };
void write(unsigned address, byte value) override {};
void clear(byte value) override {};
private:
unsigned size;
};
class DivPlatformOPL: public DivDispatch {
protected:
struct Channel: public SharedChannel<int> {
DivInstrumentFM state;
2024-07-11 20:16:41 +09:00
unsigned int freqH, freqL;
int sample, fixedFreq;
2024-07-11 20:16:41 +09:00
bool furnacePCM, fourOp, hardReset, writeCtrl;
bool levelDirect, damp, pseudoReverb;
int pan;
int macroVolMul;
Channel():
SharedChannel<int>(0),
freqH(0),
freqL(0),
2022-05-13 16:52:44 -05:00
sample(-1),
2022-05-16 21:06:49 -05:00
fixedFreq(0),
2022-05-13 16:52:44 -05:00
furnacePCM(false),
2022-03-14 02:30:25 -05:00
fourOp(false),
2022-05-07 21:48:29 -05:00
hardReset(false),
2024-07-11 20:16:41 +09:00
writeCtrl(false),
levelDirect(true),
damp(false),
pseudoReverb(false),
pan(3),
macroVolMul(64) {
2022-03-16 01:53:46 -05:00
state.ops=2;
}
};
Channel chan[44];
DivDispatchOscBuffer* oscBuf[44];
bool isMuted[44];
struct QueuedWrite {
unsigned short addr;
unsigned char val;
bool addrOrVal;
2023-07-13 04:09:20 -05:00
QueuedWrite(): addr(0), val(0), addrOrVal(false) {}
QueuedWrite(unsigned short a, unsigned char v): addr(a), val(v), addrOrVal(false) {}
};
FixedQueue<QueuedWrite,4096> writes;
2023-11-23 18:28:39 -05:00
unsigned int dacVal;
2023-11-25 02:26:59 -05:00
unsigned int dacVal2;
2023-11-23 18:28:39 -05:00
int dacOut;
2023-11-25 02:26:59 -05:00
int dacOut3[4];
2023-11-23 18:28:39 -05:00
bool lastSH;
2023-11-25 02:26:59 -05:00
bool lastSH2;
2023-11-23 18:28:39 -05:00
bool lastSY;
bool waitingBusy;
int downsamplerStep;
2023-11-22 19:28:36 -05:00
2022-05-13 14:59:36 -05:00
unsigned char* adpcmBMem;
size_t adpcmBMemLen;
unsigned char* pcmMem;
size_t pcmMemLen;
2022-05-13 14:59:36 -05:00
DivOPLAInterface iface;
DivYMF278MemoryInterface pcmMemory;
unsigned int sampleOffB[256];
unsigned int sampleOffPCM[256];
bool sampleLoaded[256];
2022-05-13 14:59:36 -05:00
ymfm::adpcm_b_engine* adpcmB;
2022-03-06 18:10:12 -05:00
const unsigned char** slotsNonDrums;
const unsigned char** slotsDrums;
const unsigned char** slots;
2022-03-07 17:07:29 -05:00
const unsigned short* chanMap;
const unsigned char* outChanMap;
int chipFreqBase, chipRateBase;
int delay, chipType, oplType, chans, melodicChans, totalChans, adpcmChan=-1, pcmChanOffs=-1, sampleBank, totalOutputs, ramSize;
unsigned char lastBusy;
2022-03-16 01:53:46 -05:00
unsigned char drumState;
unsigned char drumVol[5];
unsigned char regPool[768];
2022-03-16 01:53:46 -05:00
bool properDrums, properDrumsSys, dam, dvb;
2022-03-07 17:07:29 -05:00
unsigned char lfoValue;
2023-11-22 19:28:36 -05:00
// 0: Nuked-OPL3
// 1: ymfm
// 2: YM3812-LLE/YMF262-LLE
unsigned char emuCore;
bool update4OpMask, pretendYMU, downsample, compatPan;
short oldWrites[768];
short pendingWrites[768];
2023-11-22 19:28:36 -05:00
// chips
opl3_chip fm;
YMF278 pcm;
2023-11-22 19:28:36 -05:00
ymfm::ym3526* fm_ymfm1;
ymfm::ym3812* fm_ymfm2;
ymfm::y8950* fm_ymfm8950;
ymfm::ymf262* fm_ymfm3;
ymfm::ymf278b* fm_ymfm4;
2023-11-22 19:28:36 -05:00
fmopl2_t fm_lle2;
fmopl3_t fm_lle3;
2024-03-05 18:55:18 -05:00
DivMemoryComposition memCompo;
int octave(int freq);
int toFreq(int freq);
2022-05-13 16:52:44 -05:00
double NOTE_ADPCMB(int note);
2023-01-19 02:45:02 -05:00
void commitState(int ch, DivInstrument* ins);
friend void putDispatchChip(void*,int);
friend void putDispatchChan(void*,int,int);
2023-11-22 19:28:36 -05:00
void acquire_nukedLLE2(short** buf, size_t len);
void acquire_nukedLLE3(short** buf, size_t len);
2023-01-03 01:09:46 -05:00
void acquire_nuked(short** buf, size_t len);
2023-11-22 19:28:36 -05:00
void acquire_ymfm3(short** buf, size_t len);
void acquire_ymfm4(short** buf, size_t len);
2023-11-22 19:28:36 -05:00
void acquire_ymfm8950(short** buf, size_t len);
void acquire_ymfm2(short** buf, size_t len);
void acquire_ymfm1(short** buf, size_t len);
public:
2023-01-02 04:53:37 -05:00
void acquire(short** buf, size_t len);
int dispatch(DivCommand c);
void* getChanState(int chan);
DivMacroInt* getChanMacroInt(int ch);
2023-08-23 17:25:05 -05:00
unsigned short getPan(int chan);
DivChannelPair getPaired(int chan);
DivDispatchOscBuffer* getOscBuffer(int chan);
2023-12-17 14:54:38 -05:00
int mapVelocity(int ch, float vel);
unsigned char* getRegisterPool();
int getRegisterPoolSize();
void reset();
void forceIns();
void tick(bool sysTick=true);
void muteChannel(int ch, bool mute);
int getOutputCount();
2023-11-22 19:28:36 -05:00
void setCore(unsigned char which);
2022-03-07 17:07:29 -05:00
void setOPLType(int type, bool drums);
bool keyOffAffectsArp(int ch);
bool keyOffAffectsPorta(int ch);
bool getLegacyAlwaysSetVolume();
void toggleRegisterDump(bool enable);
void setFlags(const DivConfig& flags);
void notifyInsChange(int ins);
void notifyInsDeletion(void* ins);
int getPortaFloor(int ch);
void poke(unsigned int addr, unsigned short val);
void poke(std::vector<DivRegWrite>& wlist);
2022-05-13 14:59:36 -05:00
const void* getSampleMem(int index);
size_t getSampleMemCapacity(int index);
size_t getSampleMemUsage(int index);
bool isSampleLoaded(int index, int sample);
2024-03-05 18:55:18 -05:00
const DivMemoryComposition* getMemCompo(int index);
void renderSamples(int chipID);
int init(DivEngine* parent, int channels, int sugRate, const DivConfig& flags);
void quit();
YMF278& getChip();
DivPlatformOPL():
pcmMemory(0x400000),
pcm(pcmMemory) {}
~DivPlatformOPL();
};
#endif