Wire envelope amplitude shaping.

Signed-off-by: Adrian.Conlon <adrian.conlon@gmail.com>
This commit is contained in:
Adrian.Conlon 2017-09-23 00:07:30 +01:00
parent 09f7fe5dfe
commit 5084d85e49
2 changed files with 37 additions and 19 deletions

View File

@ -19,20 +19,24 @@ namespace EightBit {
void reset(); void reset();
bool zeroed() const; bool zeroed() const;
int default() const; int volume() const;
void setDefault(int value); void setVolume(int value);
Direction direction() const; Direction direction() const;
void setDirection(int value); void setDirection(int value);
void setDirection(Direction value); void setDirection(Direction value);
int stepLength() const; int period() const;
void setStepLength(int value); void setPeriod(int value);
void step();
private: private:
int m_defaultValue; int m_volume;
int m_direction; int m_direction;
int m_stepLength; int m_period;
int m_position;
}; };
class Sweep final { class Sweep final {

View File

@ -6,19 +6,19 @@
EightBit::GameBoy::Envelope::Envelope() {} EightBit::GameBoy::Envelope::Envelope() {}
void EightBit::GameBoy::Envelope::reset() { void EightBit::GameBoy::Envelope::reset() {
m_defaultValue = m_direction = m_stepLength = 0; m_position = m_volume = m_direction = m_period = 0;
} }
bool EightBit::GameBoy::Envelope::zeroed() const { bool EightBit::GameBoy::Envelope::zeroed() const {
return (default() == 0) && (stepLength() == 0) && (direction() == Attenuate); return (volume() == 0) && (period() == 0) && (direction() == Attenuate);
} }
int EightBit::GameBoy::Envelope::default() const { int EightBit::GameBoy::Envelope::volume() const {
return m_defaultValue; return m_volume;
} }
void EightBit::GameBoy::Envelope::setDefault(int value) { void EightBit::GameBoy::Envelope::setVolume(int value) {
m_defaultValue = value; m_volume = value;
} }
EightBit::GameBoy::Envelope::Direction EightBit::GameBoy::Envelope::direction() const { EightBit::GameBoy::Envelope::Direction EightBit::GameBoy::Envelope::direction() const {
@ -33,12 +33,23 @@ void EightBit::GameBoy::Envelope::setDirection(Direction value) {
setDirection((int)value); setDirection((int)value);
} }
int EightBit::GameBoy::Envelope::stepLength() const { int EightBit::GameBoy::Envelope::period() const {
return m_stepLength; return m_period;
} }
void EightBit::GameBoy::Envelope::setStepLength(int value) { void EightBit::GameBoy::Envelope::setPeriod(int value) {
m_stepLength = value; m_position = m_period = value;
}
void EightBit::GameBoy::Envelope::step() {
if (m_period != 0) {
if (--m_position == 0) {
auto volume = m_volume + (m_direction == Amplify ? +1 : -1);
if (volume >= 0 || volume <= 15)
m_volume = volume;
m_position == m_period;
}
}
} }
// //
@ -495,15 +506,15 @@ void EightBit::GameBoy::Audio::fromNRx1(int i, uint8_t value) {
uint8_t EightBit::GameBoy::Audio::toNRx2(int i) { uint8_t EightBit::GameBoy::Audio::toNRx2(int i) {
auto voice = (EnvelopedRectangularVoice*)m_voices[i].get(); auto voice = (EnvelopedRectangularVoice*)m_voices[i].get();
auto& envelope = voice->envelope(); auto& envelope = voice->envelope();
return (envelope.default() << 4) | (envelope.direction() << 3) | envelope.stepLength(); return (envelope.volume() << 4) | (envelope.direction() << 3) | envelope.period();
} }
void EightBit::GameBoy::Audio::fromNRx2(int i, uint8_t value) { void EightBit::GameBoy::Audio::fromNRx2(int i, uint8_t value) {
auto voice = (EnvelopedRectangularVoice*)m_voices[i].get(); auto voice = (EnvelopedRectangularVoice*)m_voices[i].get();
auto& envelope = voice->envelope(); auto& envelope = voice->envelope();
envelope.setDefault((value >> 4) & Processor::Mask4); // Bits 4-7 envelope.setVolume((value >> 4) & Processor::Mask4); // Bits 4-7
envelope.setDirection((value >> 3) & Processor::Mask1); // Bit 3 envelope.setDirection((value >> 3) & Processor::Mask1); // Bit 3
envelope.setStepLength(value & Processor::Mask3); // Bits 0-2 envelope.setPeriod(value & Processor::Mask3); // Bits 0-2
} }
uint8_t EightBit::GameBoy::Audio::toNRx3(int i) { uint8_t EightBit::GameBoy::Audio::toNRx3(int i) {
@ -776,6 +787,9 @@ void EightBit::GameBoy::Audio::Sequencer_LengthStep(const int step) {
} }
void EightBit::GameBoy::Audio::Sequencer_VolumeStep(const int step) { void EightBit::GameBoy::Audio::Sequencer_VolumeStep(const int step) {
voice1()->envelope().step();
voice2()->envelope().step();
voice4()->envelope().step();
} }
void EightBit::GameBoy::Audio::Sequencer_SweepStep(const int step) { void EightBit::GameBoy::Audio::Sequencer_SweepStep(const int step) {