mirror of
https://github.com/libretro/Mesen.git
synced 2024-12-13 11:57:09 +00:00
79 lines
1.6 KiB
C++
79 lines
1.6 KiB
C++
#pragma once
|
|
#include "stdafx.h"
|
|
#include "Snapshotable.h"
|
|
|
|
class Vrc6Saw : public Snapshotable
|
|
{
|
|
private:
|
|
uint8_t _accumulatorRate = 0;
|
|
uint8_t _accumulator = 0;
|
|
uint16_t _frequency = 1;
|
|
bool _enabled = false;
|
|
|
|
int32_t _timer = 1;
|
|
uint8_t _step = 0;
|
|
uint8_t _frequencyShift = 0;
|
|
|
|
void StreamState(bool saving) override
|
|
{
|
|
Stream(_accumulatorRate, _accumulator, _frequency, _enabled, _timer, _step, _frequencyShift);
|
|
}
|
|
|
|
public:
|
|
void WriteReg(uint16_t addr, uint8_t value)
|
|
{
|
|
switch(addr & 0x03) {
|
|
case 0:
|
|
_accumulatorRate = value & 0x3F;
|
|
break;
|
|
|
|
case 1:
|
|
_frequency = (_frequency & 0x0F00) | value;
|
|
break;
|
|
|
|
case 2:
|
|
_frequency = (_frequency & 0xFF) | ((value & 0x0F) << 8);
|
|
_enabled = (value & 0x80) == 0x80;
|
|
if(!_enabled) {
|
|
//If E is clear, the accumulator is forced to zero until E is again set.
|
|
_accumulator = 0;
|
|
|
|
//"The phase of the saw generator can be mostly reset by clearing and immediately setting E. Clearing E does not reset the frequency divider, however."
|
|
_step = 0;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void SetFrequencyShift(uint8_t shift)
|
|
{
|
|
_frequencyShift = shift;
|
|
}
|
|
|
|
void Clock()
|
|
{
|
|
if(_enabled) {
|
|
_timer--;
|
|
if(_timer == 0) {
|
|
_step = (_step + 1) % 14;
|
|
_timer = (_frequency >> _frequencyShift) + 1;
|
|
|
|
if(_step == 0) {
|
|
_accumulator = 0;
|
|
} else if((_step & 0x01) == 0x00) {
|
|
_accumulator += _accumulatorRate;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
uint8_t GetVolume()
|
|
{
|
|
if(!_enabled) {
|
|
return 0;
|
|
} else {
|
|
//"The high 5 bits of the accumulator are then output (provided the channel is enabled by having the E bit set)."
|
|
return _accumulator >> 3;
|
|
}
|
|
}
|
|
}; |