2021-07-22 21:16:23 -04:00
|
|
|
//
|
|
|
|
// Chipset.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 22/07/2021.
|
|
|
|
// Copyright © 2021 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef Chipset_hpp
|
|
|
|
#define Chipset_hpp
|
|
|
|
|
2021-10-21 20:48:57 -07:00
|
|
|
#include <algorithm>
|
2021-10-08 18:11:47 -07:00
|
|
|
#include <array>
|
2021-10-21 20:48:57 -07:00
|
|
|
#include <cassert>
|
2021-07-22 21:16:23 -04:00
|
|
|
#include <cstddef>
|
|
|
|
#include <cstdint>
|
|
|
|
|
2021-10-04 06:44:54 -07:00
|
|
|
#include "../../Activity/Source.hpp"
|
2021-10-05 15:38:56 -07:00
|
|
|
#include "../../ClockReceiver/ClockingHintSource.hpp"
|
2021-11-11 09:24:15 -05:00
|
|
|
#include "../../ClockReceiver/JustInTime.hpp"
|
2021-10-04 06:44:54 -07:00
|
|
|
#include "../../Components/6526/6526.hpp"
|
2021-07-26 18:44:20 -04:00
|
|
|
#include "../../Outputs/CRT/CRT.hpp"
|
2021-10-04 06:44:54 -07:00
|
|
|
#include "../../Processors/68000/68000.hpp"
|
2021-10-04 16:45:05 -07:00
|
|
|
#include "../../Storage/Disk/Controller/DiskController.hpp"
|
2021-10-04 08:12:13 -07:00
|
|
|
#include "../../Storage/Disk/Drive.hpp"
|
2021-07-22 21:16:23 -04:00
|
|
|
|
2021-11-09 07:11:23 -05:00
|
|
|
#include "Audio.hpp"
|
2021-11-26 18:16:24 -05:00
|
|
|
#include "Bitplanes.hpp"
|
2021-07-22 21:16:23 -04:00
|
|
|
#include "Blitter.hpp"
|
2021-09-16 21:17:23 -04:00
|
|
|
#include "Copper.hpp"
|
|
|
|
#include "DMADevice.hpp"
|
2021-10-13 15:09:19 -07:00
|
|
|
#include "Flags.hpp"
|
2021-11-06 12:03:09 -07:00
|
|
|
#include "Keyboard.hpp"
|
2021-11-26 17:50:47 -05:00
|
|
|
#include "MouseJoystick.hpp"
|
2021-10-04 06:44:54 -07:00
|
|
|
#include "MemoryMap.hpp"
|
2021-11-26 15:30:31 -05:00
|
|
|
#include "Sprites.hpp"
|
2021-07-22 21:16:23 -04:00
|
|
|
|
|
|
|
namespace Amiga {
|
|
|
|
|
2021-10-05 15:38:56 -07:00
|
|
|
class Chipset: private ClockingHint::Observer {
|
2021-07-22 21:16:23 -04:00
|
|
|
public:
|
2021-10-04 08:12:13 -07:00
|
|
|
Chipset(MemoryMap &memory_map, int input_clock_rate);
|
2021-07-22 21:16:23 -04:00
|
|
|
|
2021-07-23 21:24:07 -04:00
|
|
|
struct Changes {
|
2021-07-26 16:40:42 -04:00
|
|
|
int interrupt_level = 0;
|
2021-07-26 20:13:06 -04:00
|
|
|
HalfCycles duration;
|
2021-08-08 21:52:28 -04:00
|
|
|
|
|
|
|
Changes &operator += (const Changes &rhs) {
|
|
|
|
duration += rhs.duration;
|
|
|
|
return *this;
|
|
|
|
}
|
2021-07-23 21:24:07 -04:00
|
|
|
};
|
|
|
|
|
2021-07-27 16:41:18 -04:00
|
|
|
/// Advances the stated amount of time.
|
|
|
|
Changes run_for(HalfCycles);
|
|
|
|
|
|
|
|
/// Advances to the next available CPU slot.
|
|
|
|
Changes run_until_cpu_slot();
|
2021-07-22 21:16:23 -04:00
|
|
|
|
|
|
|
/// Performs the provided microcycle, which the caller guarantees to be a memory access.
|
|
|
|
void perform(const CPU::MC68000::Microcycle &);
|
|
|
|
|
2021-07-28 19:36:30 -04:00
|
|
|
/// Sets the current state of the CIA interrupt lines.
|
|
|
|
void set_cia_interrupts(bool cia_a, bool cia_b);
|
|
|
|
|
2021-07-26 16:40:42 -04:00
|
|
|
/// Provides the chipset's current interrupt level.
|
|
|
|
int get_interrupt_level() {
|
|
|
|
return interrupt_level_;
|
|
|
|
}
|
|
|
|
|
2021-10-05 16:12:30 -07:00
|
|
|
/// Inserts the disks provided.
|
|
|
|
/// @returns @c true if anything was inserted; @c false otherwise.
|
|
|
|
bool insert(const std::vector<std::shared_ptr<Storage::Disk::Disk>> &disks);
|
|
|
|
|
2021-07-26 18:44:20 -04:00
|
|
|
// The standard CRT set.
|
|
|
|
void set_scan_target(Outputs::Display::ScanTarget *scan_target);
|
|
|
|
Outputs::Display::ScanStatus get_scaled_scan_status() const;
|
|
|
|
void set_display_type(Outputs::Display::DisplayType);
|
|
|
|
Outputs::Display::DisplayType get_display_type() const;
|
|
|
|
|
2021-10-04 06:44:54 -07:00
|
|
|
// Activity observation.
|
|
|
|
void set_activity_observer(Activity::Observer *observer) {
|
|
|
|
cia_a_handler_.set_activity_observer(observer);
|
2021-10-06 04:54:40 -07:00
|
|
|
disk_controller_.set_activity_observer(observer);
|
2021-10-04 06:44:54 -07:00
|
|
|
}
|
|
|
|
|
2021-11-17 15:33:46 -05:00
|
|
|
// Keyboard and mouse exposure.
|
2021-11-06 12:03:09 -07:00
|
|
|
Keyboard &get_keyboard() {
|
|
|
|
return keyboard_;
|
|
|
|
}
|
|
|
|
|
2021-11-17 15:33:46 -05:00
|
|
|
const std::vector<std::unique_ptr<Inputs::Joystick>> &get_joysticks() {
|
|
|
|
return joysticks_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Synchronisation.
|
2021-11-12 15:30:52 -05:00
|
|
|
void flush();
|
|
|
|
|
2021-11-26 18:16:24 -05:00
|
|
|
// Input for receiving collected bitplanes.
|
|
|
|
void post_bitplanes(const BitplaneData &data);
|
|
|
|
|
2021-12-01 05:37:58 -05:00
|
|
|
// Obtains the source of audio output.
|
|
|
|
Outputs::Speaker::Speaker *get_speaker() {
|
|
|
|
return audio_.get_speaker();
|
|
|
|
}
|
|
|
|
|
2021-07-22 21:16:23 -04:00
|
|
|
private:
|
2021-09-14 20:51:32 -04:00
|
|
|
friend class DMADeviceBase;
|
2021-08-09 17:35:09 -04:00
|
|
|
|
2021-11-07 05:19:16 -08:00
|
|
|
// MARK: - E Clock and keyboard dividers.
|
2021-10-04 06:44:54 -07:00
|
|
|
|
|
|
|
HalfCycles cia_divider_;
|
2021-11-07 05:19:16 -08:00
|
|
|
HalfCycles keyboard_divider_;
|
2021-10-04 06:44:54 -07:00
|
|
|
|
2021-07-22 21:16:23 -04:00
|
|
|
// MARK: - Interrupts.
|
|
|
|
|
|
|
|
uint16_t interrupt_enable_ = 0;
|
|
|
|
uint16_t interrupt_requests_ = 0;
|
2021-07-26 16:40:42 -04:00
|
|
|
int interrupt_level_ = 0;
|
2021-07-22 21:16:23 -04:00
|
|
|
|
2021-07-26 16:40:42 -04:00
|
|
|
void update_interrupts();
|
2021-08-09 20:31:14 -04:00
|
|
|
void posit_interrupt(InterruptFlag);
|
2021-07-22 21:16:23 -04:00
|
|
|
|
2021-07-27 16:41:18 -04:00
|
|
|
// MARK: - Scheduler.
|
|
|
|
|
2021-08-08 21:52:28 -04:00
|
|
|
template <bool stop_on_cpu> Changes run(HalfCycles duration = HalfCycles::max());
|
2021-07-27 21:59:27 -04:00
|
|
|
template <bool stop_on_cpu> int advance_slots(int, int);
|
2021-07-27 16:41:18 -04:00
|
|
|
template <int cycle, bool stop_if_cpu> bool perform_cycle();
|
2021-07-27 21:33:07 -04:00
|
|
|
template <int cycle> void output();
|
2021-11-28 05:06:30 -05:00
|
|
|
void output_pixels(int cycles_until_sync);
|
|
|
|
void apply_ham(uint8_t);
|
2021-07-27 16:41:18 -04:00
|
|
|
|
|
|
|
// MARK: - DMA Control, Scheduler and Blitter.
|
2021-07-22 21:16:23 -04:00
|
|
|
|
|
|
|
uint16_t dma_control_ = 0;
|
|
|
|
Blitter blitter_;
|
|
|
|
|
2021-11-27 15:03:46 -05:00
|
|
|
// MARK: - Sprites and collision flags.
|
2021-07-22 21:16:23 -04:00
|
|
|
|
2021-11-26 16:02:18 -05:00
|
|
|
std::array<Sprite, 8> sprites_;
|
|
|
|
std::array<TwoSpriteShifter, 4> sprite_shifters_;
|
2021-11-27 15:03:46 -05:00
|
|
|
uint16_t collisions_ = 0, collisions_flags_= 0;
|
|
|
|
|
|
|
|
uint32_t playfield_collision_mask_ = 0, playfield_collision_complement_ = 0;
|
2021-10-25 16:30:30 -07:00
|
|
|
|
2021-08-10 19:01:41 -04:00
|
|
|
// MARK: - Raster position and state.
|
2021-07-22 21:45:51 -04:00
|
|
|
|
2021-08-10 19:01:41 -04:00
|
|
|
// Definitions related to PAL/NTSC.
|
2021-11-02 14:34:03 -07:00
|
|
|
// (Default values are PAL).
|
2021-07-27 21:33:07 -04:00
|
|
|
int line_length_ = 227;
|
2021-11-02 14:34:03 -07:00
|
|
|
int short_field_height_ = 312;
|
2021-11-26 09:37:52 -05:00
|
|
|
int vertical_blank_height_ = 25; // PAL = 25, NTSC = 20
|
2021-07-22 22:00:53 -04:00
|
|
|
|
2021-08-10 19:01:41 -04:00
|
|
|
// Current raster position.
|
|
|
|
int line_cycle_ = 0, y_ = 0;
|
|
|
|
|
|
|
|
// Parameters affecting bitplane collection and output.
|
2021-07-22 22:00:53 -04:00
|
|
|
uint16_t display_window_start_[2] = {0, 0};
|
|
|
|
uint16_t display_window_stop_[2] = {0, 0};
|
|
|
|
uint16_t fetch_window_[2] = {0, 0};
|
2021-07-26 18:44:20 -04:00
|
|
|
|
2021-08-10 19:01:41 -04:00
|
|
|
// Ephemeral bitplane collection state.
|
|
|
|
bool fetch_vertical_ = false, fetch_horizontal_ = false;
|
2021-08-11 20:31:37 -04:00
|
|
|
bool display_horizontal_ = false;
|
2021-08-10 21:28:48 -04:00
|
|
|
bool did_fetch_ = false;
|
2021-10-31 09:01:38 -07:00
|
|
|
uint16_t fetch_stop_ = 0xffff;
|
2021-08-10 19:01:41 -04:00
|
|
|
|
2021-08-11 20:31:37 -04:00
|
|
|
// Output state.
|
|
|
|
uint16_t border_colour_ = 0;
|
|
|
|
bool is_border_ = true;
|
|
|
|
int zone_duration_ = 0;
|
|
|
|
uint16_t *pixels_ = nullptr;
|
2021-11-28 05:06:30 -05:00
|
|
|
uint16_t last_colour_ = 0; // Retained for HAM mode.
|
2021-08-11 20:31:37 -04:00
|
|
|
void flush_output();
|
|
|
|
|
2021-11-26 18:16:24 -05:00
|
|
|
Bitplanes bitplanes_;
|
2021-08-10 21:28:48 -04:00
|
|
|
|
2021-11-01 14:18:58 -07:00
|
|
|
BitplaneData next_bitplanes_, previous_bitplanes_;
|
|
|
|
bool has_next_bitplanes_ = false;
|
2021-10-21 20:48:57 -07:00
|
|
|
|
2021-11-01 17:15:36 -07:00
|
|
|
int odd_priority_ = 0, even_priority_ = 0;
|
|
|
|
bool even_over_odd_ = false;
|
|
|
|
bool hold_and_modify_ = false;
|
|
|
|
bool dual_playfields_ = false;
|
|
|
|
bool interlace_ = false;
|
2021-11-02 03:47:39 -07:00
|
|
|
bool is_long_field_ = false;
|
2021-11-01 17:15:36 -07:00
|
|
|
|
2021-11-26 18:16:24 -05:00
|
|
|
BitplaneShifter bitplane_pixels_;
|
2021-08-10 19:01:41 -04:00
|
|
|
|
2021-08-11 18:47:35 -04:00
|
|
|
int odd_delay_ = 0, even_delay_ = 0;
|
2021-09-20 19:00:52 -04:00
|
|
|
bool is_high_res_ = false;
|
2021-08-11 18:47:35 -04:00
|
|
|
|
2021-07-26 18:44:20 -04:00
|
|
|
// MARK: - Copper.
|
|
|
|
|
2021-09-16 21:17:23 -04:00
|
|
|
Copper copper_;
|
2021-07-30 18:24:27 -04:00
|
|
|
|
2021-11-09 07:11:23 -05:00
|
|
|
// MARK: - Audio.
|
|
|
|
|
2021-11-13 15:53:41 -05:00
|
|
|
Audio audio_;
|
2021-11-09 07:11:23 -05:00
|
|
|
|
2021-07-30 18:24:27 -04:00
|
|
|
// MARK: - Serial port.
|
|
|
|
|
|
|
|
class SerialPort {
|
|
|
|
public:
|
|
|
|
void set_control(uint16_t) {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint16_t value = 0, reload = 0;
|
|
|
|
uint16_t shift = 0, receive_shift = 0;
|
|
|
|
uint16_t status;
|
|
|
|
} serial_;
|
2021-07-26 18:44:20 -04:00
|
|
|
|
2021-10-09 04:08:59 -07:00
|
|
|
// MARK: - Pixel output.
|
|
|
|
|
|
|
|
Outputs::CRT::CRT crt_;
|
|
|
|
uint16_t palette_[32]{};
|
2021-11-01 14:44:30 -07:00
|
|
|
uint16_t swizzled_palette_[64]{};
|
2021-10-09 04:08:59 -07:00
|
|
|
|
2021-10-23 20:17:13 -07:00
|
|
|
// MARK: - Mouse.
|
|
|
|
private:
|
2021-11-26 17:50:47 -05:00
|
|
|
Mouse mouse_;
|
2021-10-23 20:17:13 -07:00
|
|
|
|
|
|
|
public:
|
2021-11-26 17:50:47 -05:00
|
|
|
Inputs::Mouse &get_mouse() {
|
|
|
|
return mouse_;
|
|
|
|
}
|
2021-10-23 20:17:13 -07:00
|
|
|
|
2021-11-17 14:26:51 -05:00
|
|
|
// MARK: - Joystick.
|
|
|
|
private:
|
2021-11-17 15:33:46 -05:00
|
|
|
std::vector<std::unique_ptr<Inputs::Joystick>> joysticks_;
|
|
|
|
Joystick &joystick(size_t index) const {
|
|
|
|
return *static_cast<Joystick *>(joysticks_[index].get());
|
|
|
|
}
|
|
|
|
|
2021-10-23 20:17:13 -07:00
|
|
|
// MARK: - CIAs.
|
2021-10-09 04:08:59 -07:00
|
|
|
private:
|
|
|
|
class DiskController;
|
|
|
|
|
|
|
|
class CIAAHandler: public MOS::MOS6526::PortHandler {
|
|
|
|
public:
|
2021-10-23 20:17:13 -07:00
|
|
|
CIAAHandler(MemoryMap &map, DiskController &controller, Mouse &mouse);
|
2021-10-09 04:08:59 -07:00
|
|
|
void set_port_output(MOS::MOS6526::Port port, uint8_t value);
|
|
|
|
uint8_t get_port_input(MOS::MOS6526::Port port);
|
|
|
|
void set_activity_observer(Activity::Observer *observer);
|
|
|
|
|
2021-11-17 15:33:46 -05:00
|
|
|
// TEMPORARY.
|
|
|
|
// TODO: generalise mice and joysticks.
|
|
|
|
// This is a hack. A TEMPORARY HACK.
|
|
|
|
void set_joystick(Joystick *joystick) {
|
|
|
|
joystick_ = joystick;
|
|
|
|
}
|
|
|
|
|
2021-10-09 04:08:59 -07:00
|
|
|
private:
|
|
|
|
MemoryMap &map_;
|
|
|
|
DiskController &controller_;
|
2021-10-23 20:17:13 -07:00
|
|
|
Mouse &mouse_;
|
2021-11-17 15:33:46 -05:00
|
|
|
Joystick *joystick_ = nullptr;
|
2021-10-09 04:08:59 -07:00
|
|
|
Activity::Observer *observer_ = nullptr;
|
|
|
|
inline static const std::string led_name = "Power";
|
|
|
|
} cia_a_handler_;
|
|
|
|
|
|
|
|
class CIABHandler: public MOS::MOS6526::PortHandler {
|
|
|
|
public:
|
|
|
|
CIABHandler(DiskController &controller);
|
|
|
|
void set_port_output(MOS::MOS6526::Port port, uint8_t value);
|
|
|
|
uint8_t get_port_input(MOS::MOS6526::Port);
|
|
|
|
|
|
|
|
private:
|
|
|
|
DiskController &controller_;
|
|
|
|
} cia_b_handler_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
using CIAA = MOS::MOS6526::MOS6526<CIAAHandler, MOS::MOS6526::Personality::P8250>;
|
|
|
|
using CIAB = MOS::MOS6526::MOS6526<CIABHandler, MOS::MOS6526::Personality::P8250>;
|
|
|
|
|
|
|
|
// CIAs are provided for direct access; it's up to the caller properly
|
|
|
|
// to distinguish relevant accesses.
|
|
|
|
CIAA cia_a;
|
|
|
|
CIAB cia_b;
|
|
|
|
|
|
|
|
private:
|
2021-08-09 17:35:09 -04:00
|
|
|
// MARK: - Disk drives.
|
|
|
|
|
2021-10-08 17:18:11 -07:00
|
|
|
class DiskDMA: public DMADevice<1> {
|
|
|
|
public:
|
|
|
|
using DMADevice::DMADevice;
|
|
|
|
|
2021-11-02 18:18:59 -07:00
|
|
|
void set_length(uint16_t);
|
|
|
|
void set_control(uint16_t);
|
2021-11-13 15:53:41 -05:00
|
|
|
bool advance_dma();
|
2021-10-08 17:18:11 -07:00
|
|
|
|
|
|
|
void enqueue(uint16_t value, bool matches_sync);
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint16_t length_;
|
|
|
|
bool dma_enable_ = false;
|
|
|
|
bool write_ = false;
|
2021-10-11 06:16:01 -07:00
|
|
|
uint16_t last_set_length_ = 0;
|
2021-11-02 18:18:59 -07:00
|
|
|
bool sync_with_word_ = false;
|
2021-10-08 18:11:47 -07:00
|
|
|
|
|
|
|
std::array<uint16_t, 4> buffer_;
|
|
|
|
size_t buffer_read_ = 0, buffer_write_ = 0;
|
2021-11-02 18:18:59 -07:00
|
|
|
|
|
|
|
enum class State {
|
|
|
|
Inactive,
|
|
|
|
WaitingForSync,
|
|
|
|
Reading,
|
|
|
|
} state_ = State::Inactive;
|
2021-10-08 17:18:11 -07:00
|
|
|
} disk_;
|
|
|
|
|
2021-10-05 15:38:56 -07:00
|
|
|
class DiskController: public Storage::Disk::Controller {
|
2021-10-04 16:45:05 -07:00
|
|
|
public:
|
2021-10-14 16:36:17 -07:00
|
|
|
DiskController(Cycles clock_rate, Chipset &chipset, DiskDMA &disk_dma, CIAB &cia);
|
2021-10-04 16:45:05 -07:00
|
|
|
|
2021-10-05 05:12:01 -07:00
|
|
|
void set_mtr_sel_side_dir_step(uint8_t);
|
|
|
|
uint8_t get_rdy_trk0_wpro_chng();
|
2021-10-04 16:45:05 -07:00
|
|
|
|
2021-10-05 15:38:56 -07:00
|
|
|
void run_for(Cycles duration) {
|
|
|
|
Storage::Disk::Controller::run_for(duration);
|
|
|
|
}
|
|
|
|
|
2021-10-05 16:12:30 -07:00
|
|
|
bool insert(const std::shared_ptr<Storage::Disk::Disk> &disk, size_t drive);
|
2021-10-06 04:54:40 -07:00
|
|
|
void set_activity_observer(Activity::Observer *);
|
2021-10-05 16:12:30 -07:00
|
|
|
|
2021-10-07 05:11:32 -07:00
|
|
|
void set_sync_word(uint16_t);
|
|
|
|
void set_control(uint16_t);
|
|
|
|
|
2021-10-04 16:45:05 -07:00
|
|
|
private:
|
|
|
|
void process_input_bit(int value) final;
|
|
|
|
void process_index_hole() final;
|
|
|
|
|
2021-10-05 05:12:01 -07:00
|
|
|
// Implement the Amiga's drive ID shift registers
|
|
|
|
// directly in the controller for now.
|
|
|
|
uint32_t drive_ids_[4]{};
|
|
|
|
uint32_t previous_select_ = 0;
|
|
|
|
|
2021-10-07 05:11:32 -07:00
|
|
|
uint16_t data_ = 0;
|
|
|
|
int bit_count_ = 0;
|
2021-10-14 16:36:17 -07:00
|
|
|
uint16_t sync_word_ = 0x4489; // TODO: confirm or deny guess.
|
2021-10-07 05:11:32 -07:00
|
|
|
bool sync_with_word_ = false;
|
|
|
|
|
2021-10-14 16:36:17 -07:00
|
|
|
Chipset &chipset_;
|
2021-10-08 17:18:11 -07:00
|
|
|
DiskDMA &disk_dma_;
|
2021-10-09 04:08:59 -07:00
|
|
|
CIAB &cia_;
|
2021-10-08 17:18:11 -07:00
|
|
|
|
2021-10-04 16:45:05 -07:00
|
|
|
} disk_controller_;
|
2021-10-14 16:36:17 -07:00
|
|
|
friend DiskController;
|
2021-10-07 05:11:32 -07:00
|
|
|
|
2021-10-05 15:38:56 -07:00
|
|
|
void set_component_prefers_clocking(ClockingHint::Source *, ClockingHint::Preference) final;
|
|
|
|
bool disk_controller_is_sleeping_ = false;
|
2021-10-07 05:11:32 -07:00
|
|
|
uint16_t paula_disk_control_ = 0;
|
2021-11-06 12:03:09 -07:00
|
|
|
|
|
|
|
// MARK: - Keyboard.
|
|
|
|
|
|
|
|
Keyboard keyboard_;
|
2021-07-22 21:16:23 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* Chipset_hpp */
|