2021-07-18 11:36:13 -04:00
|
|
|
//
|
|
|
|
// 6526.h
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 18/07/2021.
|
|
|
|
// Copyright © 2021 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef _526_h
|
|
|
|
#define _526_h
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
|
2021-07-18 12:23:47 -04:00
|
|
|
#include "Implementation/6526Storage.hpp"
|
2021-11-04 18:54:28 -07:00
|
|
|
#include "../Serial/Line.hpp"
|
2021-07-18 12:23:47 -04:00
|
|
|
|
2021-07-18 11:36:13 -04:00
|
|
|
namespace MOS {
|
|
|
|
namespace MOS6526 {
|
|
|
|
|
2021-07-18 17:17:41 -04:00
|
|
|
enum Port {
|
|
|
|
A = 0,
|
|
|
|
B = 1
|
|
|
|
};
|
2021-07-18 11:36:13 -04:00
|
|
|
|
2021-07-18 17:17:41 -04:00
|
|
|
struct PortHandler {
|
2021-07-18 20:25:04 -04:00
|
|
|
/// Requests the current input value of @c port from the port handler.
|
|
|
|
uint8_t get_port_input([[maybe_unused]] Port port) {
|
|
|
|
return 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the current output value of @c port; any bits marked as input will be supplied as 1s.
|
2021-07-18 17:17:41 -04:00
|
|
|
void set_port_output([[maybe_unused]] Port port, [[maybe_unused]] uint8_t value) {}
|
2021-07-18 11:36:13 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
enum class Personality {
|
|
|
|
// The 6526, used in machines such as the C64, has a BCD time-of-day clock.
|
2021-07-18 11:49:10 -04:00
|
|
|
P6526,
|
2021-07-18 11:36:13 -04:00
|
|
|
// The 8250, used in the Amiga, provides a binary time-of-day clock.
|
2021-07-18 11:49:10 -04:00
|
|
|
P8250,
|
2021-07-18 11:36:13 -04:00
|
|
|
};
|
|
|
|
|
2021-07-18 12:23:47 -04:00
|
|
|
template <typename PortHandlerT, Personality personality> class MOS6526:
|
2021-11-07 05:18:54 -08:00
|
|
|
private MOS6526Storage,
|
|
|
|
private Serial::Line<true>::ReadDelegate
|
2021-07-18 12:23:47 -04:00
|
|
|
{
|
2021-07-18 11:36:13 -04:00
|
|
|
public:
|
2021-11-07 05:18:54 -08:00
|
|
|
MOS6526(PortHandlerT &port_handler) noexcept : port_handler_(port_handler) {
|
|
|
|
serial_input.set_read_delegate(this);
|
|
|
|
}
|
2021-07-18 11:49:10 -04:00
|
|
|
MOS6526(const MOS6526 &) = delete;
|
|
|
|
|
2021-07-18 11:36:13 -04:00
|
|
|
/// Writes @c value to the register at @c address. Only the low two bits of the address are decoded.
|
|
|
|
void write(int address, uint8_t value);
|
|
|
|
|
|
|
|
/// Fetches the value of the register @c address. Only the low two bits of the address are decoded.
|
|
|
|
uint8_t read(int address);
|
2021-07-18 11:49:10 -04:00
|
|
|
|
2021-07-23 21:24:07 -04:00
|
|
|
/// Pulses Phi2 to advance by the specified number of half cycles.
|
2021-07-18 11:49:10 -04:00
|
|
|
void run_for(const HalfCycles half_cycles);
|
|
|
|
|
2021-07-23 21:24:07 -04:00
|
|
|
/// Pulses the TOD input the specified number of times.
|
|
|
|
void advance_tod(int count);
|
|
|
|
|
2021-07-28 19:36:30 -04:00
|
|
|
/// @returns @c true if the interrupt output is active, @c false otherwise.
|
|
|
|
bool get_interrupt_line();
|
|
|
|
|
2021-08-03 22:19:41 -04:00
|
|
|
/// Sets the current state of the CNT input.
|
|
|
|
void set_cnt_input(bool active);
|
|
|
|
|
2021-11-06 16:54:20 -07:00
|
|
|
/// Provides both the serial input bit and an additional source of CNT.
|
|
|
|
Serial::Line<true> serial_input;
|
2021-11-04 18:54:28 -07:00
|
|
|
|
2021-10-09 04:08:59 -07:00
|
|
|
/// Sets the current state of the FLG input.
|
|
|
|
void set_flag_input(bool low);
|
|
|
|
|
2021-07-18 11:49:10 -04:00
|
|
|
private:
|
|
|
|
PortHandlerT &port_handler_;
|
2021-08-03 18:50:58 -04:00
|
|
|
TODStorage<personality == Personality::P8250> tod_;
|
2021-07-18 17:17:41 -04:00
|
|
|
|
|
|
|
template <int port> void set_port_output();
|
2021-07-18 20:25:04 -04:00
|
|
|
template <int port> uint8_t get_port_input();
|
2021-07-22 16:09:32 -04:00
|
|
|
void update_interrupts();
|
2021-07-23 21:58:52 -04:00
|
|
|
void posit_interrupt(uint8_t mask);
|
2021-08-01 18:14:10 -04:00
|
|
|
void advance_counters(int);
|
2021-11-07 05:18:54 -08:00
|
|
|
|
|
|
|
bool serial_line_did_produce_bit(Serial::Line<true> *line, int bit) final;
|
2021-07-18 11:36:13 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "Implementation/6526Implementation.hpp"
|
|
|
|
|
|
|
|
#endif /* _526_h */
|