2020-10-31 20:39:32 -04:00
|
|
|
|
//
|
|
|
|
|
// Video.hpp
|
|
|
|
|
// Clock Signal
|
|
|
|
|
//
|
|
|
|
|
// Created by Thomas Harte on 31/10/2020.
|
|
|
|
|
// Copyright © 2020 Thomas Harte. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#ifndef Apple_IIgs_Video_hpp
|
|
|
|
|
#define Apple_IIgs_Video_hpp
|
|
|
|
|
|
|
|
|
|
#include "../AppleII/VideoSwitches.hpp"
|
2020-11-07 20:42:34 -05:00
|
|
|
|
#include "../../../Outputs/CRT/CRT.hpp"
|
2020-11-05 17:56:20 -05:00
|
|
|
|
#include "../../../ClockReceiver/ClockReceiver.hpp"
|
2020-10-31 20:39:32 -04:00
|
|
|
|
|
2023-05-10 16:02:18 -05:00
|
|
|
|
namespace Apple::IIgs::Video {
|
2020-10-31 20:39:32 -04:00
|
|
|
|
|
2020-11-07 19:40:26 -05:00
|
|
|
|
/*!
|
2020-11-26 12:54:20 -05:00
|
|
|
|
Provides IIgs video output; assumed clocking here is seven times the usual Apple II clock.
|
|
|
|
|
So it'll produce a single line of video every 456 cycles — 65*7 + 1, allowing for the
|
2020-11-07 19:40:26 -05:00
|
|
|
|
stretched cycle.
|
|
|
|
|
*/
|
2020-11-26 12:54:20 -05:00
|
|
|
|
class Video: public Apple::II::VideoSwitches<Cycles> {
|
2020-10-31 20:39:32 -04:00
|
|
|
|
public:
|
2020-11-26 12:54:20 -05:00
|
|
|
|
Video();
|
2020-11-05 18:17:21 -05:00
|
|
|
|
void set_internal_ram(const uint8_t *);
|
2020-10-31 20:39:32 -04:00
|
|
|
|
|
2020-11-16 21:55:41 -05:00
|
|
|
|
bool get_is_vertical_blank(Cycles offset);
|
2020-11-29 19:57:35 -05:00
|
|
|
|
uint8_t get_horizontal_counter(Cycles offset);
|
|
|
|
|
uint8_t get_vertical_counter(Cycles offset);
|
2020-11-05 17:56:20 -05:00
|
|
|
|
|
|
|
|
|
void set_new_video(uint8_t);
|
|
|
|
|
uint8_t get_new_video();
|
|
|
|
|
|
|
|
|
|
void clear_interrupts(uint8_t);
|
|
|
|
|
uint8_t get_interrupt_register();
|
|
|
|
|
void set_interrupt_register(uint8_t);
|
2020-11-29 21:21:46 -05:00
|
|
|
|
bool get_interrupt_line();
|
2020-11-05 17:56:20 -05:00
|
|
|
|
|
|
|
|
|
void notify_clock_tick();
|
2020-11-07 23:14:50 -05:00
|
|
|
|
|
2020-11-07 22:03:05 -05:00
|
|
|
|
void set_border_colour(uint8_t);
|
2020-11-07 23:14:50 -05:00
|
|
|
|
void set_text_colour(uint8_t);
|
2020-12-11 21:43:34 -05:00
|
|
|
|
uint8_t get_text_colour();
|
2020-11-26 13:13:48 -05:00
|
|
|
|
uint8_t get_border_colour();
|
2020-11-05 17:56:20 -05:00
|
|
|
|
|
2020-11-09 21:54:25 -05:00
|
|
|
|
void set_composite_is_colour(bool);
|
|
|
|
|
bool get_composite_is_colour();
|
|
|
|
|
|
2020-11-07 20:42:34 -05:00
|
|
|
|
/// Sets the scan target.
|
|
|
|
|
void set_scan_target(Outputs::Display::ScanTarget *scan_target);
|
|
|
|
|
|
|
|
|
|
/// Gets the current scan status.
|
|
|
|
|
Outputs::Display::ScanStatus get_scaled_scan_status() const;
|
|
|
|
|
|
|
|
|
|
/// Sets the type of output.
|
|
|
|
|
void set_display_type(Outputs::Display::DisplayType);
|
|
|
|
|
|
|
|
|
|
/// Gets the type of output.
|
|
|
|
|
Outputs::Display::DisplayType get_display_type() const;
|
|
|
|
|
|
2020-11-16 14:42:50 -05:00
|
|
|
|
/// Determines the period until video might autonomously update its interrupt lines.
|
2023-09-10 18:00:49 -04:00
|
|
|
|
Cycles next_sequence_point() const;
|
2020-11-16 14:42:50 -05:00
|
|
|
|
|
2020-11-29 21:21:46 -05:00
|
|
|
|
/// Sets the Mega II interrupt enable state — 1/4-second and VBL interrupts are
|
|
|
|
|
/// generated here.
|
|
|
|
|
void set_megaii_interrupts_enabled(uint8_t);
|
|
|
|
|
|
|
|
|
|
uint8_t get_megaii_interrupt_status();
|
|
|
|
|
|
|
|
|
|
void clear_megaii_interrupts();
|
|
|
|
|
|
2020-10-31 20:39:32 -04:00
|
|
|
|
private:
|
2020-11-07 20:42:34 -05:00
|
|
|
|
Outputs::CRT::CRT crt_;
|
|
|
|
|
|
2020-11-26 12:54:20 -05:00
|
|
|
|
// This is coupled to Apple::II::GraphicsMode, but adds detail for the IIgs.
|
|
|
|
|
enum class GraphicsMode {
|
|
|
|
|
Text = 0,
|
|
|
|
|
DoubleText,
|
|
|
|
|
HighRes,
|
|
|
|
|
DoubleHighRes,
|
|
|
|
|
LowRes,
|
|
|
|
|
DoubleLowRes,
|
|
|
|
|
FatLowRes,
|
|
|
|
|
|
|
|
|
|
// Additions:
|
|
|
|
|
DoubleHighResMono,
|
|
|
|
|
SuperHighRes
|
|
|
|
|
};
|
|
|
|
|
constexpr bool is_colour_ntsc(GraphicsMode m) { return m >= GraphicsMode::HighRes && m <= GraphicsMode::FatLowRes; }
|
|
|
|
|
|
2020-11-22 21:29:40 -05:00
|
|
|
|
GraphicsMode graphics_mode(int row) const {
|
|
|
|
|
if(new_video_ & 0x80) {
|
|
|
|
|
return GraphicsMode::SuperHighRes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const auto ii_mode = Apple::II::VideoSwitches<Cycles>::graphics_mode(row);
|
|
|
|
|
switch(ii_mode) {
|
|
|
|
|
// Coupling very much assumed here.
|
|
|
|
|
case Apple::II::GraphicsMode::DoubleHighRes:
|
|
|
|
|
if(new_video_ & 0x20) {
|
|
|
|
|
return GraphicsMode::DoubleHighResMono;
|
|
|
|
|
}
|
|
|
|
|
[[fallthrough]];
|
|
|
|
|
|
|
|
|
|
default: return GraphicsMode(int(ii_mode)); break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-26 12:54:20 -05:00
|
|
|
|
enum class PixelBufferFormat {
|
|
|
|
|
Text, DoubleText, NTSC, NTSCMono, SuperHighRes
|
|
|
|
|
};
|
|
|
|
|
constexpr PixelBufferFormat format_for_mode(GraphicsMode m) {
|
|
|
|
|
switch(m) {
|
|
|
|
|
case GraphicsMode::Text: return PixelBufferFormat::Text;
|
|
|
|
|
case GraphicsMode::DoubleText: return PixelBufferFormat::DoubleText;
|
2023-05-12 14:14:45 -04:00
|
|
|
|
default: return PixelBufferFormat::NTSC;
|
2020-11-26 12:54:20 -05:00
|
|
|
|
case GraphicsMode::DoubleHighResMono: return PixelBufferFormat::NTSCMono;
|
|
|
|
|
case GraphicsMode::SuperHighRes: return PixelBufferFormat::SuperHighRes;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-07 17:45:03 -05:00
|
|
|
|
void advance(Cycles);
|
2020-11-05 17:56:20 -05:00
|
|
|
|
|
|
|
|
|
uint8_t new_video_ = 0x01;
|
2023-08-18 14:30:40 -04:00
|
|
|
|
|
|
|
|
|
class Interrupts {
|
|
|
|
|
public:
|
|
|
|
|
void add(uint8_t value) {
|
|
|
|
|
// Taken literally, status accumulates regardless of being enabled,
|
|
|
|
|
// potentially to be polled, it simply doesn't trigger an interrupt.
|
|
|
|
|
value_ |= value;
|
|
|
|
|
test();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void clear(uint8_t value) {
|
|
|
|
|
// Zeroes in bits 5 or 6 clear the respective interrupts.
|
|
|
|
|
value_ &= value | ~0x60;
|
|
|
|
|
test();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void set_control(uint8_t value) {
|
|
|
|
|
// Ones in bits 1 or 2 enable the respective interrupts.
|
|
|
|
|
value_ = (value_ & ~0x6) | (value & 0x6);
|
|
|
|
|
test();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t status() const {
|
|
|
|
|
return value_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool active() const {
|
|
|
|
|
return value_ & 0x80;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void test() {
|
|
|
|
|
value_ &= 0x7f;
|
|
|
|
|
if((value_ >> 4) & value_ & 0x6) {
|
|
|
|
|
value_ |= 0x80;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Overall meaning of value is as per the VGC interrupt register, i.e.
|
|
|
|
|
//
|
|
|
|
|
// b7: interrupt status;
|
|
|
|
|
// b6: 1-second interrupt status;
|
|
|
|
|
// b5: scan-line interrupt status;
|
|
|
|
|
// b4: reserved;
|
|
|
|
|
// b3: reserved;
|
|
|
|
|
// b2: 1-second interrupt enable;
|
|
|
|
|
// b1: scan-line interrupt enable;
|
|
|
|
|
// b0: reserved.
|
|
|
|
|
uint8_t value_ = 0x00;
|
|
|
|
|
} interrupts_;
|
2020-11-05 17:56:20 -05:00
|
|
|
|
|
|
|
|
|
int cycles_into_frame_ = 0;
|
2020-11-05 18:17:21 -05:00
|
|
|
|
const uint8_t *ram_ = nullptr;
|
2020-11-07 23:14:50 -05:00
|
|
|
|
|
|
|
|
|
// The modal colours.
|
2020-11-07 22:03:05 -05:00
|
|
|
|
uint16_t border_colour_ = 0;
|
2020-11-26 13:13:48 -05:00
|
|
|
|
uint8_t border_colour_entry_ = 0;
|
2020-12-11 21:43:34 -05:00
|
|
|
|
uint8_t text_colour_entry_ = 0xf0;
|
2020-11-08 17:04:04 -05:00
|
|
|
|
uint16_t text_colour_ = 0xffff;
|
2020-11-07 23:14:50 -05:00
|
|
|
|
uint16_t background_colour_ = 0;
|
2020-11-07 20:42:34 -05:00
|
|
|
|
|
2020-11-22 21:29:40 -05:00
|
|
|
|
// Current pixel output buffer and conceptual format.
|
|
|
|
|
PixelBufferFormat pixels_format_;
|
2020-11-08 17:01:23 -05:00
|
|
|
|
uint16_t *pixels_ = nullptr, *next_pixel_ = nullptr;
|
2020-11-22 21:29:40 -05:00
|
|
|
|
int pixels_start_column_;
|
2020-11-08 17:01:23 -05:00
|
|
|
|
|
2020-11-07 20:42:34 -05:00
|
|
|
|
void output_row(int row, int start, int end);
|
2020-11-15 17:16:52 -05:00
|
|
|
|
|
|
|
|
|
uint16_t *output_super_high_res(uint16_t *target, int start, int end, int row) const;
|
2020-11-18 19:47:22 -05:00
|
|
|
|
|
2020-11-15 17:16:52 -05:00
|
|
|
|
uint16_t *output_text(uint16_t *target, int start, int end, int row) const;
|
2020-11-18 19:47:22 -05:00
|
|
|
|
uint16_t *output_double_text(uint16_t *target, int start, int end, int row) const;
|
2020-11-18 21:06:18 -05:00
|
|
|
|
uint16_t *output_char(uint16_t *target, uint8_t source, int row) const;
|
2020-11-15 18:36:24 -05:00
|
|
|
|
|
2020-11-21 18:07:51 -05:00
|
|
|
|
uint16_t *output_low_resolution(uint16_t *target, int start, int end, int row);
|
2020-11-23 20:58:32 -05:00
|
|
|
|
uint16_t *output_fat_low_resolution(uint16_t *target, int start, int end, int row);
|
2020-11-23 19:05:18 -05:00
|
|
|
|
uint16_t *output_double_low_resolution(uint16_t *target, int start, int end, int row);
|
2020-11-21 18:07:51 -05:00
|
|
|
|
|
|
|
|
|
uint16_t *output_high_resolution(uint16_t *target, int start, int end, int row);
|
2020-11-23 19:05:18 -05:00
|
|
|
|
uint16_t *output_double_high_resolution(uint16_t *target, int start, int end, int row);
|
2020-11-23 20:58:32 -05:00
|
|
|
|
uint16_t *output_double_high_resolution_mono(uint16_t *target, int start, int end, int row);
|
2020-11-21 18:07:51 -05:00
|
|
|
|
|
2020-11-15 18:36:24 -05:00
|
|
|
|
// Super high-res per-line state.
|
|
|
|
|
uint8_t line_control_;
|
|
|
|
|
uint16_t palette_[16];
|
2020-11-21 18:07:51 -05:00
|
|
|
|
|
2020-11-22 21:55:21 -05:00
|
|
|
|
// Storage used for fill mode.
|
|
|
|
|
uint16_t *palette_zero_[4] = {nullptr, nullptr, nullptr, nullptr}, palette_throwaway_;
|
|
|
|
|
|
2020-11-21 18:07:51 -05:00
|
|
|
|
// Lookup tables and state to assist in the IIgs' mapping from NTSC to RGB.
|
|
|
|
|
//
|
|
|
|
|
// My understanding of the real-life algorithm is: maintain a four-bit buffer.
|
|
|
|
|
// Fill it in a circular fashion. Ordinarily, output the result of looking
|
|
|
|
|
// up the RGB mapping of those four bits of Apple II output (which outputs four
|
|
|
|
|
// bits per NTSC colour cycle), commuted as per current phase. But if the bit
|
|
|
|
|
// being inserted differs from that currently in its position in the shift
|
|
|
|
|
// register, hold the existing output for three shifts.
|
|
|
|
|
//
|
|
|
|
|
// From there I am using the following:
|
|
|
|
|
|
2020-11-22 19:10:05 -05:00
|
|
|
|
// Maps from:
|
|
|
|
|
//
|
|
|
|
|
// b0 = b0 of the shift register
|
|
|
|
|
// b1 = b4 of the shift register
|
|
|
|
|
// b2– = current delay count
|
|
|
|
|
//
|
|
|
|
|
// to a new delay count.
|
|
|
|
|
uint8_t ntsc_delay_lookup_[20];
|
|
|
|
|
uint32_t ntsc_shift_ = 0; // Assumption here: logical shifts will ensue, rather than arithmetic.
|
|
|
|
|
int ntsc_delay_ = 0;
|
2020-11-21 18:07:51 -05:00
|
|
|
|
|
|
|
|
|
/// Outputs the lowest 14 bits from @c ntsc_shift_, mapping to RGB.
|
2020-11-22 21:29:40 -05:00
|
|
|
|
/// Phase is derived from @c column.
|
|
|
|
|
uint16_t *output_shift(uint16_t *target, int column);
|
2020-11-29 19:57:35 -05:00
|
|
|
|
|
|
|
|
|
// Common getter for the two counters.
|
|
|
|
|
struct Counters {
|
|
|
|
|
Counters(int v, int h) : vertical(v), horizontal(h) {}
|
|
|
|
|
const int vertical, horizontal;
|
|
|
|
|
};
|
|
|
|
|
Counters get_counters(Cycles offset);
|
2020-11-29 21:21:46 -05:00
|
|
|
|
|
|
|
|
|
// Marshalls the Mega II-style interrupt state.
|
|
|
|
|
uint8_t megaii_interrupt_mask_ = 0;
|
|
|
|
|
uint8_t megaii_interrupt_state_ = 0;
|
|
|
|
|
int megaii_frame_counter_ = 0; // To count up to quarter-second interrupts.
|
2020-10-31 20:39:32 -04:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* Video_hpp */
|
2020-11-29 19:57:35 -05:00
|
|
|
|
|