2016-01-30 14:57:50 -05:00
|
|
|
#pragma once
|
|
|
|
#include "stdafx.h"
|
|
|
|
#include "Snapshotable.h"
|
|
|
|
|
|
|
|
class BaseFdsChannel : public Snapshotable
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
uint8_t _speed = 0;
|
|
|
|
uint8_t _gain = 0;
|
|
|
|
bool _envelopeOff = false;
|
|
|
|
bool _volumeIncrease = false;
|
|
|
|
uint16_t _frequency = 0;
|
|
|
|
|
|
|
|
uint32_t _timer = 0;
|
|
|
|
|
2016-06-25 20:46:54 -04:00
|
|
|
//"Few FDS NSFs write to this register. The BIOS initializes this to $FF."
|
|
|
|
uint8_t _masterSpeed = 0xFF;
|
2016-01-30 14:57:50 -05:00
|
|
|
|
2016-12-17 23:14:47 -05:00
|
|
|
void StreamState(bool saving) override
|
2016-01-30 14:57:50 -05:00
|
|
|
{
|
2016-06-02 20:20:26 -04:00
|
|
|
Stream(_speed, _gain, _envelopeOff, _volumeIncrease, _frequency, _timer, _masterSpeed);
|
2016-01-30 14:57:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
void SetMasterEnvelopeSpeed(uint8_t masterSpeed)
|
|
|
|
{
|
|
|
|
_masterSpeed = masterSpeed;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void WriteReg(uint16_t addr, uint8_t value)
|
|
|
|
{
|
2017-05-04 18:17:46 -04:00
|
|
|
switch(addr) {
|
|
|
|
case 0x4080:
|
2016-01-30 14:57:50 -05:00
|
|
|
_speed = value & 0x3F;
|
|
|
|
_volumeIncrease = (value & 0x40) == 0x40;
|
|
|
|
_envelopeOff = (value & 0x80) == 0x80;
|
|
|
|
|
|
|
|
//"Writing to this register immediately resets the clock timer that ticks the volume envelope (delaying the next tick slightly)."
|
2017-05-04 18:17:46 -04:00
|
|
|
ResetTimer();
|
|
|
|
|
|
|
|
if(_envelopeOff) {
|
|
|
|
//Envelope is off, gain = speed
|
|
|
|
_gain = _speed;
|
|
|
|
}
|
2016-01-30 14:57:50 -05:00
|
|
|
break;
|
|
|
|
|
2017-05-04 18:17:46 -04:00
|
|
|
case 0x4082:
|
2016-01-30 14:57:50 -05:00
|
|
|
_frequency = (_frequency & 0x0F00) | value;
|
|
|
|
break;
|
|
|
|
|
2017-05-04 18:17:46 -04:00
|
|
|
case 0x4083:
|
2016-01-30 14:57:50 -05:00
|
|
|
_frequency = (_frequency & 0xFF) | ((value & 0x0F) << 8);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-04 18:17:46 -04:00
|
|
|
bool TickEnvelope()
|
2016-01-30 14:57:50 -05:00
|
|
|
{
|
2017-05-04 18:17:46 -04:00
|
|
|
if(!_envelopeOff && _masterSpeed > 0) {
|
|
|
|
_timer--;
|
|
|
|
if(_timer == 0) {
|
|
|
|
ResetTimer();
|
2016-01-30 14:57:50 -05:00
|
|
|
|
2017-05-04 18:17:46 -04:00
|
|
|
if(_volumeIncrease && _gain < 32) {
|
|
|
|
_gain++;
|
|
|
|
} else if(!_volumeIncrease && _gain > 0) {
|
|
|
|
_gain--;
|
2016-01-30 14:57:50 -05:00
|
|
|
}
|
2017-05-04 18:17:46 -04:00
|
|
|
return true;
|
2016-01-30 14:57:50 -05:00
|
|
|
}
|
|
|
|
}
|
2017-05-04 18:17:46 -04:00
|
|
|
return false;
|
2016-01-30 14:57:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t GetGain()
|
|
|
|
{
|
|
|
|
return _gain;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t GetFrequency()
|
|
|
|
{
|
|
|
|
return _frequency;
|
|
|
|
}
|
2017-05-04 18:17:46 -04:00
|
|
|
|
|
|
|
void ResetTimer()
|
|
|
|
{
|
|
|
|
_timer = 8 * (_speed + 1) * _masterSpeed;
|
|
|
|
}
|
2016-01-30 14:57:50 -05:00
|
|
|
};
|