2021-03-23 22:01:41 +00:00
|
|
|
#pragma once
|
|
|
|
|
2023-01-28 18:15:28 +00:00
|
|
|
#include "MockingboardDefs.h"
|
2021-03-23 22:01:41 +00:00
|
|
|
|
|
|
|
class SSI263
|
|
|
|
{
|
|
|
|
public:
|
2023-01-28 18:15:28 +00:00
|
|
|
SSI263(UINT slot) : m_slot(slot)
|
2021-03-23 22:01:41 +00:00
|
|
|
{
|
|
|
|
m_device = -1; // undefined
|
|
|
|
m_cardMode = PH_Mockingboard;
|
|
|
|
m_pPhonemeData00 = NULL;
|
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
ResetState(true);
|
2021-03-23 22:01:41 +00:00
|
|
|
}
|
|
|
|
~SSI263(void)
|
|
|
|
{
|
|
|
|
delete [] m_pPhonemeData00;
|
|
|
|
}
|
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
void ResetState(const bool powerCycle)
|
2021-03-23 22:01:41 +00:00
|
|
|
{
|
|
|
|
m_currentActivePhoneme = -1;
|
|
|
|
m_isVotraxPhoneme = false;
|
2024-06-07 21:10:33 +01:00
|
|
|
m_votraxPhoneme = 0;
|
2021-03-23 22:01:41 +00:00
|
|
|
m_cyclesThisAudioFrame = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
m_lastUpdateCycle = 0;
|
|
|
|
m_updateWasFullSpeed = false;
|
|
|
|
|
|
|
|
m_pPhonemeData = NULL;
|
|
|
|
m_phonemeLengthRemaining = 0;
|
|
|
|
m_phonemeAccurateLengthRemaining = 0;
|
|
|
|
m_phonemePlaybackAndDebugger = false;
|
|
|
|
m_phonemeCompleteByFullSpeed = false;
|
2024-06-07 21:10:33 +01:00
|
|
|
m_phonemeLeadoutLength = 0;
|
2021-03-23 22:01:41 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
m_numSamplesError = 0;
|
|
|
|
m_byteOffset = (DWORD)-1;
|
|
|
|
m_currSampleSum = 0;
|
|
|
|
m_currNumSamples = 0;
|
|
|
|
m_currSampleMod4 = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
// After a chip power-on, if the first thing done is CTL=0, then empirically it can be observed that:
|
|
|
|
// . enableInts = 1 (mostly an SSI263 interrupt occurs, but not always)
|
|
|
|
// . DR1:0 != b#00 (since enableInts is set to 1, except when no interrupt => DR1:0 == b#00)
|
|
|
|
m_durationPhoneme = MODE_PHONEME_TRANSITIONED_INFLECTION; // Typical function & phoneme=$00
|
2021-03-23 22:01:41 +00:00
|
|
|
m_inflection = 0;
|
|
|
|
m_rateInflection = 0;
|
2024-06-07 21:10:33 +01:00
|
|
|
m_ctrlArtAmp = powerCycle ? CONTROL_MASK : 0; // Chip power-on, so CTL=1 (power-down / standby)
|
|
|
|
m_filterFreq = powerCycle ? FILTER_FREQ_SILENCE : 0; // Empirically observed at chip power-on (GH#1302)
|
2021-03-23 22:01:41 +00:00
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
m_currentMode.mode = 0;
|
|
|
|
m_currentMode.function = 0; // Set at runtime when CTL=0
|
|
|
|
m_currentMode.enableInts = 0; // Set at runtime when CTL=0
|
|
|
|
m_currentMode.D7 = 0; // Since in power-down mode (as per SSI263 datasheet)
|
2021-03-23 22:01:41 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
m_dbgFirst = true;
|
|
|
|
m_dbgStartTime = 0;
|
|
|
|
}
|
|
|
|
|
2023-01-28 18:15:28 +00:00
|
|
|
void SetDevice(UINT device) { m_device = device; }
|
2024-06-07 21:10:33 +01:00
|
|
|
void SetCardMode(PHASOR_MODE mode);
|
2023-01-28 18:15:28 +00:00
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
bool DSInit(void);
|
|
|
|
void DSUninit(void);
|
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
void Reset(const bool powerCycle, const bool isPhasorCard);
|
2021-03-23 22:01:41 +00:00
|
|
|
bool IsPhonemeActive(void) { return m_currentActivePhoneme >= 0; }
|
|
|
|
|
|
|
|
BYTE Read(ULONG nExecutedCycles);
|
|
|
|
void Write(BYTE nReg, BYTE nValue);
|
|
|
|
|
|
|
|
void Mute(void);
|
|
|
|
void Unmute(void);
|
|
|
|
void SetVolume(DWORD dwVolume, DWORD dwVolumeMax);
|
|
|
|
|
|
|
|
void PeriodicUpdate(UINT executedCycles);
|
|
|
|
void Update(void);
|
|
|
|
void SetSpeechIRQ(void);
|
|
|
|
|
|
|
|
void Votrax_Write(BYTE nValue);
|
|
|
|
bool GetVotraxPhoneme(void) { return m_isVotraxPhoneme; }
|
|
|
|
void SetVotraxPhoneme(bool value) { m_isVotraxPhoneme = value; }
|
|
|
|
|
|
|
|
void SaveSnapshot(class YamlSaveHelper& yamlSaveHelper);
|
2023-01-28 18:15:28 +00:00
|
|
|
void LoadSnapshot(class YamlLoadHelper& yamlLoadHelper, PHASOR_MODE mode, UINT version);
|
2024-06-07 21:10:33 +01:00
|
|
|
void SC01_SaveSnapshot(YamlSaveHelper& yamlSaveHelper);
|
|
|
|
void SC01_LoadSnapshot(YamlLoadHelper& yamlLoadHelper, UINT version);
|
2021-03-23 22:01:41 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
void Play(unsigned int nPhoneme);
|
|
|
|
void Stop(void);
|
|
|
|
void UpdateIRQ(void);
|
|
|
|
void UpdateAccurateLength(void);
|
2024-06-07 21:10:33 +01:00
|
|
|
void SetDeviceModeAndInts(void);
|
2021-03-23 22:01:41 +00:00
|
|
|
|
2023-01-28 18:15:28 +00:00
|
|
|
UINT64 GetLastCumulativeCycles(void);
|
|
|
|
void UpdateIFR(BYTE nDevice, BYTE clr_mask, BYTE set_mask);
|
|
|
|
BYTE GetPCR(BYTE nDevice);
|
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
static const BYTE m_Votrax2SSI263[/*64*/];
|
|
|
|
|
|
|
|
static const unsigned short m_kNumChannels = 1;
|
2021-07-28 12:14:22 +01:00
|
|
|
static const DWORD m_kDSBufferByteSize = MAX_SAMPLES * sizeof(short) * m_kNumChannels;
|
|
|
|
short m_mixBufferSSI263[m_kDSBufferByteSize / sizeof(short)];
|
2021-03-23 22:01:41 +00:00
|
|
|
VOICE SSI263SingleVoice;
|
|
|
|
|
|
|
|
//
|
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
// Duration/Phonome
|
|
|
|
static const BYTE DURATION_MODE_MASK = 0xC0;
|
|
|
|
static const BYTE DURATION_MODE_SHIFT = 6;
|
|
|
|
static const BYTE PHONEME_MASK = 0x3F;
|
|
|
|
|
|
|
|
static const BYTE MODE_PHONEME_TRANSITIONED_INFLECTION = 0xC0;
|
|
|
|
static const BYTE MODE_PHONEME_IMMEDIATE_INFLECTION = 0x80;
|
|
|
|
static const BYTE MODE_FRAME_IMMEDIATE_INFLECTION = 0x40;
|
|
|
|
static const BYTE MODE_IRQ_DISABLED = 0x00; // disable interrupts, but retains one of the 3 modes
|
|
|
|
|
|
|
|
// Rate/Inflection
|
|
|
|
static const BYTE RATE_MASK = 0xF0;
|
|
|
|
static const BYTE INFLECTION_MASK_H = 0x08; // I11
|
|
|
|
static const BYTE INFLECTION_MASK_L = 0x07; // I2..I0
|
|
|
|
|
|
|
|
// Ctrl/Art/Amp
|
|
|
|
static const BYTE ARTICULATION_MASK = 0x70;
|
|
|
|
static const BYTE AMPLITUDE_MASK = 0x0F;
|
|
|
|
static const BYTE CONTROL_MASK = 0x80;
|
|
|
|
|
|
|
|
// Filter frequency range
|
|
|
|
static const BYTE FILTER_FREQ_SILENCE = 0xFF;
|
|
|
|
|
2023-01-28 18:15:28 +00:00
|
|
|
UINT m_slot;
|
2021-03-23 22:01:41 +00:00
|
|
|
BYTE m_device; // SSI263 device# which is generating phoneme-complete IRQ (and only required whilst Mockingboard isn't a class)
|
|
|
|
PHASOR_MODE m_cardMode;
|
|
|
|
short* m_pPhonemeData00;
|
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
int m_currentActivePhoneme; // -1 (if none) or SSI263 or SC01 phoneme
|
2021-03-23 22:01:41 +00:00
|
|
|
bool m_isVotraxPhoneme;
|
2024-06-07 21:10:33 +01:00
|
|
|
BYTE m_votraxPhoneme;
|
2021-03-23 22:01:41 +00:00
|
|
|
UINT m_cyclesThisAudioFrame;
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
UINT64 m_lastUpdateCycle;
|
|
|
|
bool m_updateWasFullSpeed;
|
|
|
|
|
|
|
|
const short* m_pPhonemeData;
|
|
|
|
UINT m_phonemeLengthRemaining; // length in samples, decremented as space becomes available in the ring-buffer
|
|
|
|
UINT m_phonemeAccurateLengthRemaining; // length in samples, decremented by cycles executed
|
|
|
|
bool m_phonemePlaybackAndDebugger;
|
|
|
|
bool m_phonemeCompleteByFullSpeed;
|
2024-06-07 21:10:33 +01:00
|
|
|
UINT m_phonemeLeadoutLength; // length in samples, decremented after \m_phonemeLengthRemaining\ goes to zero. Delay until phoneme repeats
|
2021-03-23 22:01:41 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
int m_numSamplesError;
|
|
|
|
DWORD m_byteOffset;
|
|
|
|
int m_currSampleSum;
|
|
|
|
int m_currNumSamples;
|
|
|
|
UINT m_currSampleMod4;
|
|
|
|
|
|
|
|
// Regs:
|
|
|
|
BYTE m_durationPhoneme;
|
|
|
|
BYTE m_inflection; // I10..I3
|
|
|
|
BYTE m_rateInflection;
|
|
|
|
BYTE m_ctrlArtAmp;
|
|
|
|
BYTE m_filterFreq;
|
|
|
|
|
2024-06-07 21:10:33 +01:00
|
|
|
union
|
|
|
|
{
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
BYTE D7 : 1; // b0=D7 pin (for IRQ)
|
|
|
|
BYTE reserved : 4;
|
|
|
|
BYTE enableInts : 1; // b5 = enable A/!R (ie. interrupts)
|
|
|
|
BYTE function : 2; // b7:6 = function
|
|
|
|
};
|
|
|
|
BYTE mode;
|
|
|
|
} m_currentMode;
|
2021-03-23 22:01:41 +00:00
|
|
|
|
|
|
|
// Debug
|
|
|
|
bool m_dbgFirst;
|
|
|
|
UINT64 m_dbgStartTime;
|
|
|
|
};
|