1
0
mirror of https://github.com/TomHarte/CLK.git synced 2025-01-25 09:30:14 +00:00

184 lines
4.8 KiB
C++
Raw Normal View History

2021-06-22 19:33:41 -04:00
//
// Dave.hpp
// Clock Signal
//
// Created by Thomas Harte on 22/06/2021.
// Copyright © 2021 Thomas Harte. All rights reserved.
//
#pragma once
2021-06-22 19:33:41 -04:00
#include <cstdint>
#include "../../ClockReceiver/ClockReceiver.hpp"
#include "../../Concurrency/AsyncTaskQueue.hpp"
2021-06-22 19:33:41 -04:00
#include "../../Numeric/LFSR.hpp"
#include "../../Outputs/Speaker/Implementation/BufferSource.hpp"
2021-06-22 19:33:41 -04:00
2023-05-10 16:02:18 -05:00
namespace Enterprise::Dave {
2021-06-22 19:33:41 -04:00
enum class Interrupt: uint8_t {
VariableFrequency = 0x02,
OneHz = 0x08,
Nick = 0x20,
};
2021-06-22 19:33:41 -04:00
/*!
Models the audio-production subset of Dave's behaviour.
2021-06-22 19:33:41 -04:00
*/
class Audio: public Outputs::Speaker::BufferSource<Audio, true> {
2021-06-22 19:33:41 -04:00
public:
2022-07-16 14:41:04 -04:00
Audio(Concurrency::AsyncTaskQueue<false> &audio_queue);
2021-07-03 12:56:56 -04:00
/// Modifies an register in the audio range; only the low 4 bits are
/// used for register decoding so it's assumed that the caller has
/// already identified this write as being to an audio register.
2021-06-22 19:33:41 -04:00
void write(uint16_t address, uint8_t value);
// MARK: - SampleSource.
void set_sample_volume_range(int16_t range);
template <Outputs::Speaker::Action action>
void apply_samples(std::size_t number_of_samples, Outputs::Speaker::StereoSample *target);
2021-06-22 19:33:41 -04:00
private:
2022-07-16 14:41:04 -04:00
Concurrency::AsyncTaskQueue<false> &audio_queue_;
// Global divider (i.e. 8MHz/12Mhz switch).
uint8_t global_divider_;
uint8_t global_divider_reload_ = 2;
2021-06-26 23:48:53 -04:00
// Tone channels.
struct Channel {
// User-set values.
uint16_t reload = 0;
bool high_pass = false;
bool ring_modulate = false;
enum class Distortion {
None = 0,
FourBit = 1,
FiveBit = 2,
SevenBit = 3,
} distortion = Distortion::None;
uint8_t amplitude[2]{};
bool sync = false;
// Current state.
uint16_t count = 0;
int output = 0;
} channels_[3];
void update_channel(int);
2021-06-26 23:48:53 -04:00
// Noise channel.
struct Noise {
// User-set values.
uint8_t amplitude[2]{};
enum class Frequency {
DivideByFour,
ToneChannel0,
ToneChannel1,
ToneChannel2,
} frequency = Frequency::DivideByFour;
enum class Polynomial {
SeventeenBit,
FifteenBit,
ElevenBit,
NineBit
} polynomial = Polynomial::SeventeenBit;
bool swap_polynomial = false;
bool low_pass = false;
bool high_pass = false;
bool ring_modulate = false;
// Current state.
int count = 0;
int output = 0;
bool final_output = false;
2021-06-26 23:48:53 -04:00
} noise_;
void update_noise();
bool use_direct_output_[2]{};
2021-06-26 23:48:53 -04:00
// Global volume, per SampleSource obligations.
int16_t volume_ = 0;
2021-06-22 19:33:41 -04:00
// Polynomials that are always running.
2021-06-22 19:33:41 -04:00
Numeric::LFSRv<0xc> poly4_;
Numeric::LFSRv<0x14> poly5_;
Numeric::LFSRv<0x60> poly7_;
// The selectable, noise-related polynomial.
2021-06-22 19:33:41 -04:00
Numeric::LFSRv<0x110> poly9_;
Numeric::LFSRv<0x500> poly11_;
Numeric::LFSRv<0x6000> poly15_;
Numeric::LFSRv<0x12000> poly17_;
// Current state of the active polynomials.
uint8_t poly_state_[4];
2021-06-22 19:33:41 -04:00
};
/*!
Provides Dave's timed interrupts  those that are provided at 1 kHz,
50 Hz or according to the rate of tone generators 0 or 1, plus the fixed
1 Hz interrupt.
*/
class TimedInterruptSource {
public:
2021-07-03 12:56:56 -04:00
/// Modifies an register in the audio range; only the low 4 bits are
/// used for register decoding so it's assumed that the caller has
/// already identified this write as being to an audio register.
void write(uint16_t address, uint8_t value);
2021-07-03 12:56:56 -04:00
/// Returns a bitmask of interrupts that have become active since
/// the last time this method was called; flags are as defined in
/// @c Enterprise::Dave::Interrupt
uint8_t get_new_interrupts();
2021-07-03 12:56:56 -04:00
/// Returns the current high or low states of the inputs that trigger
/// the interrupts modelled here, as a bit mask compatible with that
/// exposed by Dave as the register at 0xb4.
uint8_t get_divider_state();
2021-07-03 12:56:56 -04:00
/// Advances the interrupt source.
void run_for(Cycles);
2021-07-03 12:56:56 -04:00
/// @returns The amount of time from now until the earliest that
/// @c get_new_interrupts() _might_ have new interrupts to report.
Cycles next_sequence_point() const;
private:
2021-06-27 21:47:21 -04:00
static constexpr Cycles clock_rate{250000};
static constexpr Cycles half_clock_rate{125000};
2021-06-27 21:47:21 -04:00
// Global divider (i.e. 8MHz/12Mhz switch).
Cycles global_divider_ = Cycles(2);
Cycles run_length_;
2021-07-03 12:56:56 -04:00
// Interrupts that have fired since get_new_interrupts()
// was last called.
uint8_t interrupts_ = 0;
// A counter for the 1Hz interrupt.
int two_second_counter_ = 0;
2021-07-03 12:56:56 -04:00
// A counter specific to the 1kHz and 50Hz timers, if in use.
enum class InterruptRate {
OnekHz,
FiftyHz,
ToneGenerator0,
ToneGenerator1,
} rate_ = InterruptRate::OnekHz;
2021-06-28 21:08:41 -04:00
bool programmable_level_ = false;
2021-07-03 12:56:56 -04:00
// A local duplicate of the counting state of the first two audio
// channels, maintained in case either of those is used as an
// interrupt source.
struct Channel {
int value = 100, reload = 100;
bool sync = false;
bool level = false;
} channels_[2];
void update_channel(int c, bool is_linked, int decrement);
};
2021-06-22 19:33:41 -04:00
}