1
0
mirror of https://github.com/TomHarte/CLK.git synced 2024-07-05 10:28:58 +00:00
CLK/Components/AY38910/AY38910.hpp

117 lines
2.8 KiB
C++
Raw Normal View History

//
// AY-3-8910.hpp
// Clock Signal
//
// Created by Thomas Harte on 14/10/2016.
// Copyright © 2016 Thomas Harte. All rights reserved.
//
#ifndef AY_3_8910_hpp
#define AY_3_8910_hpp
#include "../../Outputs/Speaker.hpp"
namespace GI {
namespace AY38910 {
class PortHandler {
public:
virtual uint8_t get_port_input(bool port_b) {
return 0xff;
}
virtual void set_port_output(bool port_b, uint8_t value) {}
};
enum ControlLines {
BC1 = (1 << 0),
BC2 = (1 << 1),
BDIR = (1 << 2)
};
2016-10-20 02:47:44 +00:00
/*!
Provides emulation of an AY-3-8910 / YM2149, which is a three-channel sound chip with a
noise generator and a volume envelope generator, which also provides two bidirectional
interface ports.
*/
class AY38910: public ::Outputs::Filter<AY38910> {
public:
2016-10-20 02:47:44 +00:00
/// Creates a new AY38910.
AY38910();
2016-10-20 02:47:44 +00:00
/// Sets the clock rate at which this AY38910 will be run.
void set_clock_rate(double clock_rate);
2016-10-20 02:47:44 +00:00
/// Sets the value the AY would read from its data lines if it were not outputting.
void set_data_input(uint8_t r);
2016-10-20 02:47:44 +00:00
/// Gets the value that would appear on the data lines if only the AY is outputting.
uint8_t get_data_output();
2016-10-20 02:47:44 +00:00
2017-08-01 21:01:36 +00:00
/// Sets the current control line state, as a bit field.
void set_control_lines(ControlLines control_lines);
2016-10-20 02:47:44 +00:00
/*!
Gets the value that would appear on the requested interface port if it were in output mode.
@parameter port_b @c true to get the value for Port B, @c false to get the value for Port A.
*/
2016-10-15 01:44:15 +00:00
uint8_t get_port_output(bool port_b);
/*!
Sets the port handler, which will receive a call every time the AY either wants to sample
input or else declare new output. As a convenience, current port output can be obtained
without installing a port handler via get_port_output.
*/
void set_port_handler(PortHandler *);
2016-10-20 02:47:44 +00:00
// to satisfy ::Outputs::Speaker (included via ::Outputs::Filter; not for public consumption
void get_samples(unsigned int number_of_samples, int16_t *target);
private:
int selected_register_;
uint8_t registers_[16], output_registers_[16];
uint8_t port_inputs_[2];
int master_divider_;
2016-10-24 00:42:49 +00:00
int tone_periods_[3];
int tone_counters_[3];
int tone_outputs_[3];
int noise_period_;
int noise_counter_;
int noise_shift_register_;
int noise_output_;
int envelope_period_;
int envelope_divider_;
int envelope_position_;
int envelope_shapes_[16][32];
int envelope_overflow_masks_[16];
int volumes_[16];
2016-10-24 00:42:49 +00:00
enum ControlState {
Inactive,
LatchAddress,
Read,
Write
} control_state_;
void select_register(uint8_t r);
void set_register_value(uint8_t value);
uint8_t get_register_value();
uint8_t data_input_, data_output_;
int16_t output_volume_;
2016-10-24 00:42:49 +00:00
inline void evaluate_output_volume();
inline void update_bus();
PortHandler *port_handler_;
};
}
}
#endif /* AY_3_8910_hpp */