2021-06-14 22:19:25 -04:00
|
|
|
//
|
|
|
|
// Nick.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 14/06/2021.
|
|
|
|
// Copyright © 2021 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef Nick_hpp
|
|
|
|
#define Nick_hpp
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
#include "../../ClockReceiver/ClockReceiver.hpp"
|
2021-06-14 23:11:48 -04:00
|
|
|
#include "../../Outputs/CRT/CRT.hpp"
|
2021-06-14 22:19:25 -04:00
|
|
|
|
|
|
|
namespace Enterprise {
|
|
|
|
|
|
|
|
class Nick {
|
|
|
|
public:
|
2021-06-15 17:43:13 -04:00
|
|
|
Nick(const uint8_t *ram);
|
2021-06-14 23:11:48 -04:00
|
|
|
|
2021-07-03 13:06:07 -04:00
|
|
|
/// Writes to a Nick register; only the low two bits are decoded.
|
2021-06-14 22:19:25 -04:00
|
|
|
void write(uint16_t address, uint8_t value);
|
2021-07-03 13:06:07 -04:00
|
|
|
|
|
|
|
/// Reads from the Nick range. Nobody seems to be completely clear what
|
|
|
|
/// this should return; I've set it up to return the last fetched video or mode
|
|
|
|
/// line byte during periods when those things are being fetched, 0xff at all
|
|
|
|
/// other times. Including during refresh, since I don't know what addresses
|
|
|
|
/// are generated then.
|
|
|
|
///
|
|
|
|
/// This likely isn't accurate, but is the most accurate guess I could make.
|
|
|
|
uint8_t read();
|
2021-06-14 22:19:25 -04:00
|
|
|
|
2021-06-17 22:30:24 -04:00
|
|
|
void run_for(Cycles);
|
2021-06-27 17:13:07 -04:00
|
|
|
Cycles get_time_until_z80_slot(Cycles after_period) const;
|
2021-06-14 23:11:48 -04:00
|
|
|
|
|
|
|
void set_scan_target(Outputs::Display::ScanTarget *scan_target);
|
|
|
|
Outputs::Display::ScanStatus get_scaled_scan_status() const;
|
|
|
|
|
2021-07-02 21:42:09 -04:00
|
|
|
/// @returns The amount of time until the next potential change in interrupt output.
|
2021-06-27 16:28:01 -04:00
|
|
|
Cycles get_next_sequence_point() const;
|
2021-06-17 22:30:24 -04:00
|
|
|
|
|
|
|
/*!
|
|
|
|
@returns The current state of the interrupt line — @c true for active;
|
|
|
|
@c false for inactive.
|
|
|
|
*/
|
2021-06-27 21:37:11 -04:00
|
|
|
inline bool get_interrupt_line() const {
|
2021-06-17 22:30:24 -04:00
|
|
|
return interrupt_line_;
|
|
|
|
}
|
|
|
|
|
2021-07-02 21:42:09 -04:00
|
|
|
/// Sets the type of output.
|
|
|
|
void set_display_type(Outputs::Display::DisplayType);
|
|
|
|
|
|
|
|
/// Gets the type of output.
|
|
|
|
Outputs::Display::DisplayType get_display_type() const;
|
|
|
|
|
2021-06-14 23:11:48 -04:00
|
|
|
private:
|
|
|
|
Outputs::CRT::CRT crt_;
|
2021-06-15 17:43:13 -04:00
|
|
|
const uint8_t *const ram_;
|
|
|
|
|
2021-06-15 18:57:14 -04:00
|
|
|
// CPU-provided state.
|
2021-06-15 17:43:13 -04:00
|
|
|
uint8_t line_parameter_control_ = 0xc0;
|
|
|
|
uint16_t line_parameter_base_ = 0x0000;
|
2021-06-15 20:55:58 -04:00
|
|
|
uint16_t border_colour_ = 0;
|
2021-06-15 18:57:14 -04:00
|
|
|
|
|
|
|
// Ephemerals, related to current video position.
|
2021-06-15 17:43:13 -04:00
|
|
|
int horizontal_counter_ = 0;
|
2021-06-15 18:57:14 -04:00
|
|
|
uint16_t line_parameter_pointer_ = 0x0000;
|
2021-06-18 17:44:17 -04:00
|
|
|
bool should_reload_line_parameters_ = true;
|
2021-06-15 20:55:58 -04:00
|
|
|
uint16_t line_data_pointer_[2];
|
2021-06-24 18:34:21 -04:00
|
|
|
uint16_t start_line_data_pointer_[2];
|
2021-07-03 13:06:07 -04:00
|
|
|
mutable uint8_t last_read_ = 0xff;
|
2021-06-15 18:57:14 -04:00
|
|
|
|
|
|
|
// Current mode line parameters.
|
|
|
|
uint8_t lines_remaining_ = 0x00;
|
2021-06-19 21:57:26 -04:00
|
|
|
uint8_t two_colour_mask_ = 0xff;
|
2021-06-15 20:55:58 -04:00
|
|
|
int left_margin_ = 0, right_margin_ = 0;
|
2021-06-19 13:04:18 -04:00
|
|
|
const uint16_t *alt_ind_palettes[4];
|
2021-06-15 20:55:58 -04:00
|
|
|
enum class Mode {
|
|
|
|
Vsync,
|
|
|
|
Pixel,
|
|
|
|
Attr,
|
|
|
|
CH256,
|
|
|
|
CH128,
|
|
|
|
CH64,
|
|
|
|
Unused,
|
|
|
|
LPixel,
|
|
|
|
} mode_ = Mode::Vsync;
|
2021-06-24 18:34:21 -04:00
|
|
|
bool is_sync_or_pixels_ = false;
|
2021-06-16 20:43:22 -04:00
|
|
|
int bpp_ = 0;
|
|
|
|
int column_size_ = 0;
|
2021-06-17 22:30:24 -04:00
|
|
|
bool interrupt_line_ = true;
|
2021-06-20 14:31:02 -04:00
|
|
|
int line_data_per_column_increments_[2] = {0, 0};
|
2021-06-24 18:34:21 -04:00
|
|
|
bool vres_ = false;
|
|
|
|
bool reload_line_parameter_pointer_ = false;
|
2021-06-15 20:55:58 -04:00
|
|
|
|
2021-06-15 22:03:41 -04:00
|
|
|
// The destination for new pixels.
|
2021-06-28 19:00:51 -04:00
|
|
|
static constexpr int allocation_size = 336;
|
2021-06-16 20:43:22 -04:00
|
|
|
static_assert((allocation_size % 16) == 0, "Allocation size must be a multiple of 16");
|
2021-06-15 22:03:41 -04:00
|
|
|
uint16_t *pixel_pointer_ = nullptr, *allocated_pointer_ = nullptr;
|
2021-06-24 18:34:21 -04:00
|
|
|
|
|
|
|
// Output transitions.
|
|
|
|
enum class OutputType {
|
|
|
|
Sync, Blank, Pixels, Border, ColourBurst
|
|
|
|
};
|
|
|
|
void set_output_type(OutputType, bool force_flush = false);
|
|
|
|
int output_duration_ = 0;
|
|
|
|
OutputType output_type_ = OutputType::Sync;
|
2021-06-16 20:43:22 -04:00
|
|
|
|
|
|
|
// Current palette.
|
|
|
|
uint16_t palette_[16]{};
|
|
|
|
|
2021-07-02 21:42:09 -04:00
|
|
|
// The first column with pixels on it; will be either 8 or 10 depending
|
|
|
|
// on whether the colour burst is meaningful to the current display type.
|
|
|
|
int first_pixel_window_ = 10;
|
|
|
|
|
2021-06-16 20:43:22 -04:00
|
|
|
// Specific outputters.
|
2021-06-28 19:00:51 -04:00
|
|
|
template <int bpp, bool is_lpixel> void output_pixel(uint16_t *target, int columns) const;
|
|
|
|
template <int bpp, int index_bits> void output_character(uint16_t *target, int columns) const;
|
|
|
|
template <int bpp> void output_attributed(uint16_t *target, int columns) const;
|
2021-06-14 22:19:25 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* Nick_hpp */
|