furnace/src/engine/dispatch.h

372 lines
8.8 KiB
C
Raw Normal View History

2022-02-15 03:12:20 +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 _DISPATCH_H
#define _DISPATCH_H
2021-12-07 02:12:16 +00:00
#include <stdlib.h>
2022-01-17 04:21:27 +00:00
#include <vector>
#define ONE_SEMITONE 2200
#define DIV_NOTE_NULL 0x7fffffff
2022-01-17 18:29:35 +00:00
#define addWrite(a,v) regWrites.push_back(DivRegWrite(a,v));
enum DivDispatchCmds {
DIV_CMD_NOTE_ON=0,
DIV_CMD_NOTE_OFF,
DIV_CMD_NOTE_OFF_ENV,
DIV_CMD_ENV_RELEASE,
DIV_CMD_INSTRUMENT,
DIV_CMD_VOLUME,
DIV_CMD_GET_VOLUME,
DIV_CMD_GET_VOLMAX,
2021-05-14 08:23:40 +00:00
DIV_CMD_NOTE_PORTA,
DIV_CMD_PITCH,
DIV_CMD_PANNING,
DIV_CMD_LEGATO,
2021-05-17 06:51:14 +00:00
DIV_CMD_PRE_PORTA,
DIV_CMD_PRE_NOTE, // used in C64
DIV_CMD_SAMPLE_MODE,
2021-12-08 23:29:50 +00:00
DIV_CMD_SAMPLE_FREQ,
2021-12-09 08:13:37 +00:00
DIV_CMD_SAMPLE_BANK,
2021-06-06 04:27:02 +00:00
DIV_CMD_FM_LFO,
DIV_CMD_FM_LFO_WAVE,
DIV_CMD_FM_TL,
DIV_CMD_FM_AR,
DIV_CMD_FM_FB,
DIV_CMD_FM_MULT,
DIV_CMD_FM_EXTCH,
2022-01-20 22:54:11 +00:00
DIV_CMD_FM_AM_DEPTH,
DIV_CMD_FM_PM_DEPTH,
DIV_CMD_GENESIS_LFO,
DIV_CMD_ARCADE_LFO,
DIV_CMD_STD_NOISE_FREQ,
DIV_CMD_STD_NOISE_MODE,
DIV_CMD_WAVE,
DIV_CMD_GB_SWEEP_TIME,
DIV_CMD_GB_SWEEP_DIR,
2021-06-09 06:08:42 +00:00
DIV_CMD_PCE_LFO_MODE,
DIV_CMD_PCE_LFO_SPEED,
2022-02-01 23:28:48 +00:00
DIV_CMD_NES_SWEEP,
DIV_CMD_C64_CUTOFF,
DIV_CMD_C64_RESONANCE,
DIV_CMD_C64_FILTER_MODE,
DIV_CMD_C64_RESET_TIME,
DIV_CMD_C64_RESET_MASK,
DIV_CMD_C64_FILTER_RESET,
DIV_CMD_C64_DUTY_RESET,
DIV_CMD_C64_EXTENDED,
DIV_CMD_C64_FINE_DUTY,
DIV_CMD_C64_FINE_CUTOFF,
2021-12-09 22:06:28 +00:00
DIV_CMD_AY_ENVELOPE_SET,
DIV_CMD_AY_ENVELOPE_LOW,
DIV_CMD_AY_ENVELOPE_HIGH,
DIV_CMD_AY_ENVELOPE_SLIDE,
DIV_CMD_AY_NOISE_MASK_AND,
DIV_CMD_AY_NOISE_MASK_OR,
DIV_CMD_AY_AUTO_ENVELOPE,
2021-12-09 22:06:28 +00:00
2022-01-15 04:26:22 +00:00
DIV_CMD_SAA_ENVELOPE,
2022-02-21 11:41:06 +00:00
DIV_CMD_LYNX_LFSR_LOAD,
2022-02-22 09:01:57 +00:00
DIV_CMD_QSOUND_ECHO_FEEDBACK,
DIV_CMD_QSOUND_ECHO_DELAY,
2022-02-22 09:01:57 +00:00
DIV_CMD_QSOUND_ECHO_LEVEL,
2022-02-21 11:41:06 +00:00
DIV_ALWAYS_SET_VOLUME,
DIV_CMD_MAX
};
struct DivCommand {
DivDispatchCmds cmd;
2022-01-08 21:03:32 +00:00
unsigned char chan, dis;
int value, value2;
DivCommand(DivDispatchCmds c, unsigned char ch, int val, int val2):
cmd(c),
chan(ch),
2022-01-08 21:03:32 +00:00
dis(ch),
value(val),
value2(val2) {}
DivCommand(DivDispatchCmds c, unsigned char ch, int val):
cmd(c),
chan(ch),
2022-01-08 21:03:32 +00:00
dis(ch),
value(val),
value2(0) {}
DivCommand(DivDispatchCmds c, unsigned char ch):
cmd(c),
chan(ch),
2022-01-08 21:03:32 +00:00
dis(ch),
value(0),
value2(0) {}
};
struct DivDelayedCommand {
int ticks;
DivCommand cmd;
};
2022-01-17 04:21:27 +00:00
struct DivRegWrite {
/**
* an address of 0xffffxx00 indicates a Furnace specific command.
2022-01-17 04:21:27 +00:00
* the following addresses are available:
* - 0xffffxx00: start sample playback
* - xx is the instance ID
2022-01-17 04:21:27 +00:00
* - data is the sample number
* - 0xffffxx01: set sample rate
* - xx is the instance ID
* - data is the sample rate
* - 0xffffxx02: stop sample playback
* - xx is the instance ID
* - 0xffffffff: reset
2022-01-17 04:21:27 +00:00
*/
unsigned int addr;
unsigned short val;
DivRegWrite(unsigned int a, unsigned short v):
2022-01-17 18:29:35 +00:00
addr(a), val(v) {}
2022-01-17 04:21:27 +00:00
};
class DivEngine;
class DivDispatch {
protected:
DivEngine* parent;
2022-01-17 04:21:27 +00:00
std::vector<DivRegWrite> regWrites;
/**
2022-01-17 04:21:27 +00:00
* please honor these variables if needed.
*/
2022-01-17 04:21:27 +00:00
bool skipRegisterWrites, dumpWrites;
public:
/**
* the rate the samples are provided.
* the engine shall resample to the output rate.
*/
int rate;
/**
* the actual chip's clock.
*/
int chipClock;
2021-12-18 08:25:42 +00:00
/**
* fill a buffer with sound data.
* @param bufL the left or mono channel buffer.
* @param bufR the right channel buffer.
* @param start the start offset.
* @param len the amount of samples to fill.
*/
virtual void acquire(short* bufL, short* bufR, size_t start, size_t len);
2021-12-18 08:25:42 +00:00
/**
* send a command to this dispatch.
* @param c a DivCommand.
* @return a return value which varies depending on the command.
*/
virtual int dispatch(DivCommand c);
2021-12-18 08:25:42 +00:00
/**
* reset the state of this dispatch.
*/
2021-12-11 08:34:43 +00:00
virtual void reset();
2021-12-18 08:25:42 +00:00
/**
* ticks this dispatch.
*/
virtual void tick();
2022-01-27 05:29:16 +00:00
/**
* get the state of a channel.
* @return a pointer, or NULL.
*/
virtual void* getChanState(int chan);
/**
* get the register pool of this dispatch.
* @return a pointer, or NULL.
*/
virtual unsigned char* getRegisterPool();
/**
* get the size of the register pool of this dispatch.
* @return the size.
*/
virtual int getRegisterPoolSize();
2022-01-27 05:29:16 +00:00
2022-02-22 09:01:57 +00:00
/**
* get the bit depth of the register pool of this dispatch.
2022-02-23 03:13:17 +00:00
* If the result is 16, it should be casted to unsigned short.
2022-02-22 09:01:57 +00:00
* @return the depth. Default value is 8
*/
virtual int getRegisterPoolDepth();
2021-12-18 08:25:42 +00:00
/**
* get this dispatch's state. DO NOT IMPLEMENT YET.
2021-12-18 08:25:42 +00:00
* @return a pointer to the dispatch's state. must be deallocated manually!
*/
virtual void* getState();
/**
* set this dispatch's state. DO NOT IMPLEMENT YET.
2021-12-18 08:25:42 +00:00
* @param state a pointer to a state pertaining to this dispatch,
* or NULL if this dispatch does not support state saves.
*/
virtual void setState(void* state);
/**
* mute a channel.
* @param ch the channel to mute.
* @param mute whether to mute or unmute.
*/
virtual void muteChannel(int ch, bool mute);
/**
* test whether this dispatch outputs audio in two channels.
* @return whether it does.
*/
virtual bool isStereo();
2021-12-18 08:25:42 +00:00
/**
* test whether sending a key off command to a channel should reset arp too.
* @param ch the channel in question.
* @return whether it does.
*/
2021-12-08 05:33:00 +00:00
virtual bool keyOffAffectsArp(int ch);
2021-12-18 08:25:42 +00:00
/**
* test whether sending a key off command to a channel should reset slides too.
* @param ch the channel in question.
* @return whether it does.
*/
virtual bool keyOffAffectsPorta(int ch);
/**
* get the lowest note in a portamento.
* @param ch the channel in question.
* @return the lowest note.
*/
virtual int getPortaFloor(int ch);
2022-01-20 18:48:20 +00:00
/**
* get a description of a dispatch-specific effect.
* @param effect the effect.
* @return the description, or NULL if effect is invalid.
*/
virtual const char* getEffectName(unsigned char effect);
2021-12-18 08:25:42 +00:00
/**
2022-01-28 17:59:53 +00:00
* set the chip flags.
* @param flags the flags. see song.h for possible values.
2021-12-18 08:25:42 +00:00
*/
2022-01-28 17:59:53 +00:00
virtual void setFlags(unsigned int flags);
/**
* set skip reg writes.
*/
void setSkipRegisterWrites(bool value);
2022-01-18 04:59:52 +00:00
/**
* notify instrument change.
*/
virtual void notifyInsChange(int ins);
2022-01-18 05:25:10 +00:00
/**
* notify wavetable change.
*/
virtual void notifyWaveChange(int wave);
2022-01-13 22:40:29 +00:00
/**
* notify deletion of an instrument.
*/
virtual void notifyInsDeletion(void* ins);
/**
* force-retrigger instruments.
*/
virtual void forceIns();
2022-01-17 04:21:27 +00:00
/**
* enable register dumping.
*/
virtual void toggleRegisterDump(bool enable);
2022-01-17 04:21:27 +00:00
/**
* get register writes.
*/
std::vector<DivRegWrite>& getRegisterWrites();
2022-02-01 23:08:19 +00:00
/**
* poke a register.
* @param addr address.
* @param val value.
*/
virtual void poke(unsigned int addr, unsigned short val);
/**
* poke a register.
* @param wlist a vector containing DivRegWrites.
*/
virtual void poke(std::vector<DivRegWrite>& wlist);
2022-02-03 23:38:57 +00:00
/**
* get available registers.
* @return an array of C strings, terminated by NULL; or NULL if none available.
*/
virtual const char** getRegisterSheet();
/**
* initialize this DivDispatch.
* @param parent the parent DivEngine.
* @param channels the number of channels to acquire.
* @param sugRate the suggested rate. this may change, so don't rely on it.
2022-01-28 17:59:53 +00:00
* @param flags the chip flags. see song.h for possible values.
* @return the number of channels allocated.
*/
2022-01-28 17:59:53 +00:00
virtual int init(DivEngine* parent, int channels, int sugRate, unsigned int flags);
/**
* quit the DivDispatch.
*/
virtual void quit();
virtual ~DivDispatch();
};
#define NOTE_PERIODIC(x) parent->calcBaseFreq(chipClock,CHIP_DIVIDER,x,true)
#define NOTE_FREQUENCY(x) parent->calcBaseFreq(chipClock,CHIP_FREQBASE,x,false)
#define COLOR_NTSC (315000000.0/88.0)
#define COLOR_PAL (283.75*15625.0+25.0)
#endif