From 5fa8e046d8c5c880a8ef41499a015d3ff66d4d51 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 27 Dec 2019 19:03:10 -0500 Subject: [PATCH 01/18] It's inaccurrate to call this _the_ shifter. So don't. --- Machines/Atari/ST/Video.cpp | 32 ++++++++++++++++---------------- Machines/Atari/ST/Video.hpp | 6 +++--- 2 files changed, 19 insertions(+), 19 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 97ccbd842..5ee0d5a2b 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -107,7 +107,7 @@ const int hsync_end = CYCLE(8); // Cycles before end of line when hsync ends. Video::Video() : deferrer_([=] (HalfCycles duration) { advance(duration); }), crt_(1024, 1, Outputs::Display::Type::PAL50, Outputs::Display::InputDataType::Red4Green4Blue4), - shifter_(crt_, palette_) { + video_stream_(crt_, palette_) { // Show a total of 260 lines; a little short for PAL but a compromise between that and the ST's // usual output height of 200 lines. @@ -201,11 +201,11 @@ void Video::advance(HalfCycles duration) { // inside the shifter on the temporary register values. if(horizontal_.sync || vertical_.sync) { - shifter_.output_sync(run_length); + video_stream_.output_sync(run_length); } else if(horizontal_.blank || vertical_.blank) { - shifter_.output_blank(run_length); + video_stream_.output_blank(run_length); } else if(!load_) { - shifter_.output_border(run_length, output_bpp_); + video_stream_.output_border(run_length, output_bpp_); } else { const int since_load = x_ - load_base_; @@ -222,12 +222,12 @@ void Video::advance(HalfCycles duration) { // was reloaded by the fetch depends on the FIFO. if(start_column == end_column) { - shifter_.output_pixels(run_length, output_bpp_); + video_stream_.output_pixels(run_length, output_bpp_); } else { // Continue the current column if partway across. if(since_load&7) { // If at least one column boundary is crossed, complete this column. - shifter_.output_pixels(8 - (since_load & 7), output_bpp_); + video_stream_.output_pixels(8 - (since_load & 7), output_bpp_); ++start_column; // This starts a new column, so latch a new word. push_latched_data(); } @@ -235,13 +235,13 @@ void Video::advance(HalfCycles duration) { // Run for all columns that have their starts in this time period. int complete_columns = end_column - start_column; while(complete_columns--) { - shifter_.output_pixels(8, output_bpp_); + video_stream_.output_pixels(8, output_bpp_); push_latched_data(); } // Output the start of the next column, if necessary. if((since_load + run_length) & 7) { - shifter_.output_pixels((since_load + run_length) & 7, output_bpp_); + video_stream_.output_pixels((since_load + run_length) & 7, output_bpp_); } } } @@ -331,7 +331,7 @@ void Video::push_latched_data() { data_latch_read_position_ = (data_latch_read_position_ + 1) & 127; if(!(data_latch_read_position_ & 3)) { - shifter_.load( + video_stream_.load( (uint64_t(data_latch_[(data_latch_read_position_ - 4) & 127]) << 48) | (uint64_t(data_latch_[(data_latch_read_position_ - 3) & 127]) << 32) | (uint64_t(data_latch_[(data_latch_read_position_ - 2) & 127]) << 16) | @@ -489,7 +489,7 @@ void Video::update_output_mode() { // MARK: - The shifter -void Video::Shifter::flush_output(OutputMode next_mode) { +void Video::VideoStream::flush_output(OutputMode next_mode) { switch(output_mode_) { case OutputMode::Sync: crt_.output_sync(duration_); break; case OutputMode::Blank: crt_.output_blank(duration_); break; @@ -513,7 +513,7 @@ void Video::Shifter::flush_output(OutputMode next_mode) { output_mode_ = next_mode; } -void Video::Shifter::output_colour_burst(int duration) { +void Video::VideoStream::output_colour_burst(int duration) { // More hackery afoot here; if and when duration_ crosses a threshold of 40, // output 40 cycles of colour burst and then redirect to blank. if(output_mode_ != OutputMode::ColourBurst) { @@ -528,7 +528,7 @@ void Video::Shifter::output_colour_burst(int duration) { } } -void Video::Shifter::output_blank(int duration) { +void Video::VideoStream::output_blank(int duration) { if(output_mode_ != OutputMode::Blank) { // Bit of a hack: if this is a transition from sync or we're really in // colour burst, divert into that. @@ -541,14 +541,14 @@ void Video::Shifter::output_blank(int duration) { duration_ += duration; } -void Video::Shifter::output_sync(int duration) { +void Video::VideoStream::output_sync(int duration) { if(output_mode_ != OutputMode::Sync) { flush_output(OutputMode::Sync); } duration_ += duration; } -void Video::Shifter::output_border(int duration, OutputBpp bpp) { +void Video::VideoStream::output_border(int duration, OutputBpp bpp) { // If there's still anything in the shifter, redirect this to an output_pixels call. if(output_shifter_) { // This doesn't take an opinion on how much of the shifter remains populated; @@ -569,7 +569,7 @@ void Video::Shifter::output_border(int duration, OutputBpp bpp) { duration_ += duration; } -void Video::Shifter::output_pixels(int duration, OutputBpp bpp) { +void Video::VideoStream::output_pixels(int duration, OutputBpp bpp) { // If the shifter is empty and there's no pixel buffer at present, // redirect this to an output_level call. Otherwise, do a quick // memset-type fill, since the special case has been detected anyway. @@ -680,7 +680,7 @@ void Video::Shifter::output_pixels(int duration, OutputBpp bpp) { } } -void Video::Shifter::load(uint64_t value) { +void Video::VideoStream::load(uint64_t value) { output_shifter_ = value; } diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index 4e240b9c1..b436602af 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -166,9 +166,9 @@ class Video { void reset_fifo(); - class Shifter { + class VideoStream { public: - Shifter(Outputs::CRT::CRT &crt, uint16_t *palette) : crt_(crt), palette_(palette) {} + VideoStream(Outputs::CRT::CRT &crt, uint16_t *palette) : crt_(crt), palette_(palette) {} void output_blank(int duration); void output_sync(int duration); void output_border(int duration, OutputBpp bpp); @@ -196,7 +196,7 @@ class Video { Outputs::CRT::CRT &crt_; uint16_t *palette_ = nullptr; - } shifter_; + } video_stream_; /// Contains copies of the various observeable fields, after the relevant propagation delay. struct PublicState { From 5026de9653e50a04c45a4caa4913c9e06750f4e4 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 27 Dec 2019 22:47:34 -0500 Subject: [PATCH 02/18] Rejigs the video stream to ensure shifter really is continuous. ... and definitively to avoid potential buffer overruns. Or, at least, to have a mechanism in place definitively to avoid them. Which will be tested and debugged as necessary. Also simplifies the colour burst and border/pixels selection logic. --- Machines/Atari/ST/Video.cpp | 357 ++++++++++++++++++++---------------- Machines/Atari/ST/Video.hpp | 50 +++-- 2 files changed, 235 insertions(+), 172 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 5ee0d5a2b..8ed5fa99d 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -201,11 +201,11 @@ void Video::advance(HalfCycles duration) { // inside the shifter on the temporary register values. if(horizontal_.sync || vertical_.sync) { - video_stream_.output_sync(run_length); + video_stream_.output(run_length, VideoStream::OutputMode::Sync); } else if(horizontal_.blank || vertical_.blank) { - video_stream_.output_blank(run_length); + video_stream_.output(run_length, VideoStream::OutputMode::Blank); } else if(!load_) { - video_stream_.output_border(run_length, output_bpp_); + video_stream_.output(run_length, VideoStream::OutputMode::Pixels); } else { const int since_load = x_ - load_base_; @@ -222,12 +222,12 @@ void Video::advance(HalfCycles duration) { // was reloaded by the fetch depends on the FIFO. if(start_column == end_column) { - video_stream_.output_pixels(run_length, output_bpp_); + video_stream_.output(run_length, VideoStream::OutputMode::Pixels); } else { // Continue the current column if partway across. if(since_load&7) { // If at least one column boundary is crossed, complete this column. - video_stream_.output_pixels(8 - (since_load & 7), output_bpp_); + video_stream_.output(8 - (since_load & 7), VideoStream::OutputMode::Pixels); ++start_column; // This starts a new column, so latch a new word. push_latched_data(); } @@ -235,13 +235,13 @@ void Video::advance(HalfCycles duration) { // Run for all columns that have their starts in this time period. int complete_columns = end_column - start_column; while(complete_columns--) { - video_stream_.output_pixels(8, output_bpp_); + video_stream_.output(8, VideoStream::OutputMode::Pixels); push_latched_data(); } // Output the start of the next column, if necessary. if((since_load + run_length) & 7) { - video_stream_.output_pixels((since_load + run_length) & 7, output_bpp_); + video_stream_.output((since_load + run_length) & 7, VideoStream::OutputMode::Pixels); } } } @@ -484,154 +484,172 @@ void Video::update_output_mode() { if(output_bpp_ != old_bpp_) { // "the 71-Hz-switch does something like a shifter-reset." (and some people use a high-low resolutions switch instead) reset_fifo(); + video_stream_.set_bpp(output_bpp_); } } // MARK: - The shifter -void Video::VideoStream::flush_output(OutputMode next_mode) { - switch(output_mode_) { - case OutputMode::Sync: crt_.output_sync(duration_); break; - case OutputMode::Blank: crt_.output_blank(duration_); break; - case OutputMode::ColourBurst: crt_.output_default_colour_burst(duration_); break; - case OutputMode::Border: { -// if(!border_colour_) { -// crt_.output_blank(duration_); -// } else { - uint16_t *const colour_pointer = reinterpret_cast(crt_.begin_data(1)); - if(colour_pointer) *colour_pointer = border_colour_; - crt_.output_level(duration_); -// } - } break; - case OutputMode::Pixels: { - crt_.output_data(duration_, pixel_pointer_); - pixel_buffer_ = nullptr; - pixel_pointer_ = 0; - } break; - } - duration_ = 0; - output_mode_ = next_mode; -} - -void Video::VideoStream::output_colour_burst(int duration) { - // More hackery afoot here; if and when duration_ crosses a threshold of 40, - // output 40 cycles of colour burst and then redirect to blank. - if(output_mode_ != OutputMode::ColourBurst) { - flush_output(OutputMode::ColourBurst); - } - duration_ += duration; - if(duration_ >= 40) { - const int blank_duration = duration_ - 40; - duration_ = 40; - flush_output(OutputMode::Blank); - output_blank(blank_duration); - } -} - -void Video::VideoStream::output_blank(int duration) { - if(output_mode_ != OutputMode::Blank) { - // Bit of a hack: if this is a transition from sync or we're really in - // colour burst, divert into that. - if(output_mode_ == OutputMode::Sync || output_mode_ == OutputMode::ColourBurst) { - output_colour_burst(duration); - return; - } - flush_output(OutputMode::Blank); - } - duration_ += duration; -} - -void Video::VideoStream::output_sync(int duration) { - if(output_mode_ != OutputMode::Sync) { - flush_output(OutputMode::Sync); - } - duration_ += duration; -} - -void Video::VideoStream::output_border(int duration, OutputBpp bpp) { - // If there's still anything in the shifter, redirect this to an output_pixels call. - if(output_shifter_) { - // This doesn't take an opinion on how much of the shifter remains populated; - // it assumes the worst case. - const int pixel_length = std::min(32, duration); - output_pixels(pixel_length, bpp); - duration -= pixel_length; - if(!duration) { - return; - } +void Video::VideoStream::output(int duration, OutputMode mode) { + // If this is a transition from sync to blank, actually transition to colour burst. + if(output_mode_ == OutputMode::Sync && mode == OutputMode::Blank) { + mode = OutputMode::ColourBurst; } - // Flush anything that isn't level output *in the current border colour*. - if(output_mode_ != OutputMode::Border || border_colour_ != palette_[0]) { - flush_output(OutputMode::Border); - border_colour_ = palette_[0]; - } - duration_ += duration; -} + // If this is seeming a transition from blank to colour burst, obey it only if/when + // sufficient colour burst has been output. + if(output_mode_ == OutputMode::Blank && mode == OutputMode::ColourBurst) { + if(duration_ + duration >= 40) { + const int overage = duration + duration_ - 40; + duration_ = 40; -void Video::VideoStream::output_pixels(int duration, OutputBpp bpp) { - // If the shifter is empty and there's no pixel buffer at present, - // redirect this to an output_level call. Otherwise, do a quick - // memset-type fill, since the special case has been detected anyway. - if(!output_shifter_) { - if(!pixel_buffer_) { - output_border(duration, bpp); + generate(overage, OutputMode::ColourBurst, true); } else { - duration_ += duration; - - switch(bpp_) { - case OutputBpp::One: { - const size_t pixels = size_t(duration << 1); - memset(&pixel_buffer_[pixel_pointer_], 0, pixels * sizeof(uint16_t)); - pixel_pointer_ += pixels; - } break; - - default: - case OutputBpp::Four: - assert(!(duration & 1)); - duration >>= 1; - case OutputBpp::Two: { - while(duration--) { - pixel_buffer_[pixel_pointer_] = palette_[0]; - ++pixel_pointer_; - } - } break; - } + mode = OutputMode::ColourBurst; } + } + + // If this is a transition, or if we're doing pixels, output whatever has been accumulated. + if(mode != output_mode_ || output_mode_ == OutputMode::Pixels) { + generate(duration, output_mode_, mode != output_mode_); + } else { + duration_ += duration; + } + + // Accumulate time in the current mode. + output_mode_ = mode; +} + +void Video::VideoStream::generate(int duration, OutputMode mode, bool is_terminal) { + // Three of these are trivial; deal with them upfront. They don't care about the duration of + // whatever is new, just about how much was accumulated prior to now. + if(mode != OutputMode::Pixels) { + switch(mode) { + default: + case OutputMode::Sync: crt_.output_sync(duration_); break; + case OutputMode::Blank: crt_.output_blank(duration_); break; + case OutputMode::ColourBurst: crt_.output_default_colour_burst(duration_); break; + } + + // Reseed duration + duration_ = duration; + + // The shifter should keep running, so throw away the proper amount of content. + shift(duration_); + return; } - // Flush anything that isn't pixel output in the proper bpp; also flush if there's nowhere - // left to put pixels. - if(output_mode_ != OutputMode::Pixels || bpp_ != bpp || pixel_pointer_ >= 320) { - flush_output(OutputMode::Pixels); - bpp_ = bpp; - pixel_buffer_ = reinterpret_cast(crt_.begin_data(320 + 32)); + // If the background colour has changed and border is accumulated, flush it. + if(border_colour_ != palette_[0] && duration_) { + flush_border(); } - duration_ += duration; + border_colour_ = palette_[0]; + // If the shifter is empty, accumulate in duration_ a promise to draw border later. + if(!output_shifter_) { + if(pixel_pointer_) { + flush_pixels(); + } + + duration_ += duration; + + // If this is terminal, we'll need to draw now. But if it isn't, job done. + if(is_terminal) { + flush_border(); + } + + return; + } + + // There's definitely some pixels to convey, but perhaps there's some border first? + if(duration_) { + flush_border(); + } + + // Time to do some pixels! + output_pixels(duration); + + // If was terminal, make sure any transient storage is output. + if(is_terminal) { + flush_pixels(); + } +} + +void Video::VideoStream::flush_border() { + // Output colour 0 for the entirety of duration_ (or black, if this is 1bpp mode). + uint16_t *const colour_pointer = reinterpret_cast(crt_.begin_data(1)); + if(colour_pointer) *colour_pointer = (bpp_ != OutputBpp::One) ? palette_[0] : 0; + crt_.output_level(duration_); + + duration_ = 0; +} + +namespace { +#if TARGET_RT_BIG_ENDIAN + constexpr int upper = 0; +#else + constexpr int upper = 1; +#endif +} + +void Video::VideoStream::shift(int duration) { switch(bpp_) { - case OutputBpp::One: { - int pixels = duration << 1; - if(pixel_buffer_) { - while(pixels--) { + case OutputBpp::One: + output_shifter_ <<= (duration << 1); + break; + case OutputBpp::Two: + while(duration--) { + shifter_halves_[upper] = (shifter_halves_[upper] << 1) & 0xfffefffe; + shifter_halves_[upper] |= (shifter_halves_[upper^1] & 0x80008000) >> 15; + shifter_halves_[upper^1] = (shifter_halves_[upper^1] << 1) & 0xfffefffe; + } + break; + case OutputBpp::Four: + while(duration) { + output_shifter_ = (output_shifter_ << 1) & 0xfffefffefffefffe; + duration -= 2; + } + break; + } +} + +// TODO: turn this into a template on current BPP, perhaps? Would avoid reevaluation of the conditional. +void Video::VideoStream::output_pixels(int duration) { + constexpr int allocation_size = 352; // i.e. 320 plus a spare 32. + + // Convert from duration to pixels. + int pixels = duration; + switch(bpp_) { + case OutputBpp::One: pixels <<= 1; break; + default: break; + case OutputBpp::Four: pixels >>= 1; break; + } + + while(pixels) { + // If no buffer is currently available, attempt to allocate one. + if(!pixel_buffer_) { + pixel_buffer_ = reinterpret_cast(crt_.begin_data(allocation_size, 2)); + + // Stop the loop if no buffer is available. + if(!pixel_buffer_) break; + } + + int pixels_to_draw = std::min(allocation_size - pixel_pointer_, pixels); + pixels -= pixels_to_draw; + + switch(bpp_) { + case OutputBpp::One: + while(pixels_to_draw--) { pixel_buffer_[pixel_pointer_] = ((output_shifter_ >> 63) & 1) * 0xffff; output_shifter_ <<= 1; + ++pixel_pointer_; } - } else { - pixel_pointer_ += size_t(pixels); - output_shifter_ <<= pixels; - } - } break; - case OutputBpp::Two: { - #if TARGET_RT_BIG_ENDIAN - constexpr int upper = 0; - #else - constexpr int upper = 1; - #endif - if(pixel_buffer_) { - while(duration--) { + break; + + case OutputBpp::Two: + while(pixels_to_draw--) { pixel_buffer_[pixel_pointer_] = palette_[ ((output_shifter_ >> 63) & 1) | ((output_shifter_ >> 46) & 2) @@ -645,20 +663,10 @@ void Video::VideoStream::output_pixels(int duration, OutputBpp bpp) { ++pixel_pointer_; } - } else { - pixel_pointer_ += size_t(duration); - while(duration--) { - shifter_halves_[upper] = (shifter_halves_[upper] << 1) & 0xfffefffe; - shifter_halves_[upper] |= (shifter_halves_[upper^1] & 0x80008000) >> 15; - shifter_halves_[upper^1] = (shifter_halves_[upper^1] << 1) & 0xfffefffe; - } - } - } break; - default: - case OutputBpp::Four: - assert(!(duration & 1)); - if(pixel_buffer_) { - while(duration) { + break; + + case OutputBpp::Four: + while(pixels_to_draw--) { pixel_buffer_[pixel_pointer_] = palette_[ ((output_shifter_ >> 63) & 1) | ((output_shifter_ >> 46) & 2) | @@ -666,18 +674,51 @@ void Video::VideoStream::output_pixels(int duration, OutputBpp bpp) { ((output_shifter_ >> 12) & 8) ]; output_shifter_ = (output_shifter_ << 1) & 0xfffefffefffefffe; + ++pixel_pointer_; - duration -= 2; } - } else { - pixel_pointer_ += size_t(duration >> 1); - while(duration) { - output_shifter_ = (output_shifter_ << 1) & 0xfffefffefffefffe; - duration -= 2; - } - } - break; + break; + } + + // Check whether the limit has been reached. + if(pixel_pointer_ == allocation_size) { + flush_pixels(); + } } + + // If duration remains, that implies no buffer was available, so + // just do the corresponding shifting and provide proper timing to the CRT. + if(pixels) { + int leftover_duration = pixels; + switch(bpp_) { + case OutputBpp::One: leftover_duration >>= 1; break; + default: break; + case OutputBpp::Four: leftover_duration <<= 1; break; + } + shift(leftover_duration); + crt_.output_data(leftover_duration); + } + + // Duration will be at most 32, since this will need to be interspersed with load() calls. + // Therefore it's always safe to assume that if a buffer is allocated, then it's big enough, + // provided it is flushed fairly proactively. +} + +void Video::VideoStream::flush_pixels() { + switch(bpp_) { + case OutputBpp::One: crt_.output_data(pixel_pointer_ >> 1, size_t(pixel_pointer_)); break; + default: crt_.output_data(pixel_pointer_); break; + case OutputBpp::Four: crt_.output_data(pixel_pointer_ << 1, size_t(pixel_pointer_)); break; + } + + pixel_pointer_ = 0; + pixel_buffer_ = nullptr; +} + +void Video::VideoStream::set_bpp(OutputBpp bpp) { + // TODO: is flushing like this correct? + output_shifter_ = 0; + bpp_ = bpp; } void Video::VideoStream::load(uint64_t value) { diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index b436602af..f6345821d 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -166,22 +166,48 @@ class Video { void reset_fifo(); + /*! + Provides a target for control over the output video stream, which is considered to be + a permanently shifting shifter, that you need to reload when appropriate, which can be + overridden by the blank and sync levels. + + This stream will automatically insert a colour burst. + */ class VideoStream { public: VideoStream(Outputs::CRT::CRT &crt, uint16_t *palette) : crt_(crt), palette_(palette) {} - void output_blank(int duration); - void output_sync(int duration); - void output_border(int duration, OutputBpp bpp); - void output_pixels(int duration, OutputBpp bpp); - void output_colour_burst(int duration); + + enum class OutputMode { + Sync, Blank, ColourBurst, Pixels, + }; + + /// Sets the current data format for the shifter. Changes in output BPP flush the shifter. + void set_bpp(OutputBpp bpp); + + void output(int duration, OutputMode mode); + + /// Loads 64 bits into the Shifter. The shifter shifts continuously. If you also declare + /// a pixels region then whatever is being shifted will reach the display, in a form that + /// depends on the current output BPP. void load(uint64_t value); private: + // The target CRT and the palette to use. + Outputs::CRT::CRT &crt_; + uint16_t *palette_ = nullptr; + + // Internal stateful processes. + void generate(int duration, OutputMode mode, bool is_terminal); + + void flush_border(); + void flush_pixels(); + void shift(int duration); + void output_pixels(int duration); + + // Internal state that is a function of output intent. int duration_ = 0; - enum class OutputMode { - Sync, Blank, Border, Pixels, ColourBurst - } output_mode_ = OutputMode::Sync; + OutputMode output_mode_ = OutputMode::Sync; uint16_t border_colour_ = 0; OutputBpp bpp_ = OutputBpp::Four; union { @@ -189,13 +215,9 @@ class Video { uint32_t shifter_halves_[2]; }; - void flush_output(OutputMode next_mode); - + // Internal state for handling output serialisation. uint16_t *pixel_buffer_ = nullptr; - size_t pixel_pointer_ = 0; - - Outputs::CRT::CRT &crt_; - uint16_t *palette_ = nullptr; + int pixel_pointer_ = 0; } video_stream_; /// Contains copies of the various observeable fields, after the relevant propagation delay. From 2757e5d60007ad348f530592760c7b1761d3fb75 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 27 Dec 2019 22:51:11 -0500 Subject: [PATCH 03/18] Removes untrue comment. --- Machines/Atari/ST/Video.cpp | 4 ---- 1 file changed, 4 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 8ed5fa99d..50c724994 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -698,10 +698,6 @@ void Video::VideoStream::output_pixels(int duration) { shift(leftover_duration); crt_.output_data(leftover_duration); } - - // Duration will be at most 32, since this will need to be interspersed with load() calls. - // Therefore it's always safe to assume that if a buffer is allocated, then it's big enough, - // provided it is flushed fairly proactively. } void Video::VideoStream::flush_pixels() { From a9d1f5d925f567214762ed6b5bd33b717ad0ae78 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 27 Dec 2019 23:47:19 -0500 Subject: [PATCH 04/18] Pulls out address reload as something I can position independently. Sadly receding it by 3 did not have the effect I was hoping for, of receding Enchanted Land's first register tweaking. --- Machines/Atari/ST/Video.cpp | 16 +++++++++++----- .../xcshareddata/xcschemes/Clock Signal.xcscheme | 2 +- 2 files changed, 12 insertions(+), 6 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 50c724994..2672fbf64 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -100,7 +100,7 @@ const int hsync_start = CYCLE(48); // Cycles before end of line when hsync star const int hsync_end = CYCLE(8); // Cycles before end of line when hsync ends. // "VSYNC starts 104 cycles after the start of the previous line's HSYNC, so that's 4 cycles before DE would be activated. "; -// hsync is at -50, so that's +54 +// hsync is at -50, so that's +54, or thereabouts. } @@ -246,6 +246,12 @@ void Video::advance(HalfCycles duration) { } } + // Check for address reload; this timing is _highly_ speculative on my part. + if(y_ == vertical_timings.height - 3 && x_ <= vsync_x_position && (x_ + run_length) > vsync_x_position) { + current_address_ = base_address_ >> 1; + reset_fifo(); // TODO: remove this, probably, once otherwise stable? + } + // Check for whether line length should have been latched during this run. if(x_ <= CYCLE(54) && (x_ + run_length) > CYCLE(54)) line_length_ = horizontal_timings.length; @@ -271,9 +277,6 @@ void Video::advance(HalfCycles duration) { } else if(y_ == 3) { next_vertical_.sync_schedule = VerticalState::SyncSchedule::End; - current_address_ = base_address_ >> 1; - reset_fifo(); // TODO: remove this, I think, once otherwise stable. - // Consider a shout out to the range observer. if(previous_base_address_ != base_address_) { previous_base_address_ = base_address_; @@ -486,6 +489,9 @@ void Video::update_output_mode() { reset_fifo(); video_stream_.set_bpp(output_bpp_); } + +// const int freqs[] = {50, 60, 72}; +// printf("%d, %d -> %d [%d %d]\n", x_ / 2, y_, freqs[int(field_frequency_)], horizontal_.enable, vertical_.enable); } // MARK: - The shifter @@ -579,7 +585,7 @@ void Video::VideoStream::generate(int duration, OutputMode mode, bool is_termina void Video::VideoStream::flush_border() { // Output colour 0 for the entirety of duration_ (or black, if this is 1bpp mode). uint16_t *const colour_pointer = reinterpret_cast(crt_.begin_data(1)); - if(colour_pointer) *colour_pointer = (bpp_ != OutputBpp::One) ? palette_[0] : 0; + if(colour_pointer) *colour_pointer = (bpp_ != OutputBpp::One) ? border_colour_ : 0; crt_.output_level(duration_); duration_ = 0; diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme index 175c1d515..7b1a6914b 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -67,7 +67,7 @@ Date: Sat, 28 Dec 2019 10:36:50 -0500 Subject: [PATCH 05/18] Pulls out address reload position as a separate constant. --- Machines/Atari/ST/Video.cpp | 42 ++++++++++++++++++++----------------- 1 file changed, 23 insertions(+), 19 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 2672fbf64..a97d9b0ae 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -94,10 +94,11 @@ struct Checker { } checker; #endif -const int de_delay_period = CYCLE(28); // Number of half cycles after DE that observed DE changes. -const int vsync_x_position = CYCLE(54); // Horizontal cycle on which vertical sync changes happen. -const int hsync_start = CYCLE(48); // Cycles before end of line when hsync starts. -const int hsync_end = CYCLE(8); // Cycles before end of line when hsync ends. +const int de_delay_period = CYCLE(28); // Number of half cycles after DE that observed DE changes. +const int vsync_x_position = CYCLE(54); // Horizontal cycle on which vertical sync changes happen. +const int address_reload_x_position = CYCLE(54); // Horizontal cycle on which the base address is reloaded. +const int hsync_start = CYCLE(48); // Cycles before end of line when hsync starts. +const int hsync_end = CYCLE(8); // Cycles before end of line when hsync ends. // "VSYNC starts 104 cycles after the start of the previous line's HSYNC, so that's 4 cycles before DE would be activated. "; // hsync is at -50, so that's +54, or thereabouts. @@ -246,10 +247,21 @@ void Video::advance(HalfCycles duration) { } } - // Check for address reload; this timing is _highly_ speculative on my part. - if(y_ == vertical_timings.height - 3 && x_ <= vsync_x_position && (x_ + run_length) > vsync_x_position) { + // Check for address reload; this timing is _highly_ speculative on my part. I originally had it at frame end, + // then Enchanted Woods seemed to be doing its things three lines too late, so I moved it forward a little. + // Which didn't solve the problem, so I guess that title's logic isn't triggered on address reload, but it makes + // sense to keep it out separate and I've no better intel about its actual positioning. + if(y_ == vertical_timings.height - 3 && x_ <= address_reload_x_position && (x_ + run_length) > address_reload_x_position) { current_address_ = base_address_ >> 1; reset_fifo(); // TODO: remove this, probably, once otherwise stable? + + // Consider a shout out to the range observer. + if(previous_base_address_ != base_address_) { + previous_base_address_ = base_address_; + if(range_observer_) { + range_observer_->video_did_change_access_range(this); + } + } } // Check for whether line length should have been latched during this run. @@ -270,20 +282,12 @@ void Video::advance(HalfCycles duration) { next_vertical_.enable = true; } else if(y_ == vertical_timings.reset_enable) { next_vertical_.enable = false; + } else if(next_y_ == vertical_timings.height - 1) { + next_vertical_.sync_schedule = VerticalState::SyncSchedule::Begin; } else if(next_y_ == vertical_timings.height) { next_y_ = 0; - } else if(y_ == 0) { - next_vertical_.sync_schedule = VerticalState::SyncSchedule::Begin; - } else if(y_ == 3) { + } else if(next_y_ == 2) { next_vertical_.sync_schedule = VerticalState::SyncSchedule::End; - - // Consider a shout out to the range observer. - if(previous_base_address_ != base_address_) { - previous_base_address_ = base_address_; - if(range_observer_) { - range_observer_->video_did_change_access_range(this); - } - } } } @@ -490,8 +494,8 @@ void Video::update_output_mode() { video_stream_.set_bpp(output_bpp_); } -// const int freqs[] = {50, 60, 72}; -// printf("%d, %d -> %d [%d %d]\n", x_ / 2, y_, freqs[int(field_frequency_)], horizontal_.enable, vertical_.enable); + const int freqs[] = {50, 60, 72}; + printf("%d, %d -> %d [%d %d]\n", x_ / 2, y_, freqs[int(field_frequency_)], horizontal_.enable, vertical_.enable); } // MARK: - The shifter From 13f11e071a84e8c47b4a921aab826d955f1606b0 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 28 Dec 2019 10:45:10 -0500 Subject: [PATCH 06/18] Simplifies border colour change propagation. I'm not sure it was even technically correct as was. --- Machines/Atari/ST/Video.cpp | 21 ++++++++++++--------- Machines/Atari/ST/Video.hpp | 8 ++++++-- 2 files changed, 18 insertions(+), 11 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index a97d9b0ae..e0f01a5df 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -463,6 +463,8 @@ void Video::write(int address, uint16_t value) { case 0x24: case 0x25: case 0x26: case 0x27: case 0x28: case 0x29: case 0x2a: case 0x2b: case 0x2c: case 0x2d: case 0x2e: case 0x2f: { + if(address == 0x20) video_stream_.will_change_border_colour(); + raw_palette_[address - 0x20] = value; uint8_t *const entry = reinterpret_cast(&palette_[address - 0x20]); entry[0] = uint8_t((value & 0x700) >> 7); @@ -494,8 +496,8 @@ void Video::update_output_mode() { video_stream_.set_bpp(output_bpp_); } - const int freqs[] = {50, 60, 72}; - printf("%d, %d -> %d [%d %d]\n", x_ / 2, y_, freqs[int(field_frequency_)], horizontal_.enable, vertical_.enable); +// const int freqs[] = {50, 60, 72}; +// printf("%d, %d -> %d [%d %d]\n", x_ / 2, y_, freqs[int(field_frequency_)], horizontal_.enable, vertical_.enable); } // MARK: - The shifter @@ -550,12 +552,6 @@ void Video::VideoStream::generate(int duration, OutputMode mode, bool is_termina return; } - // If the background colour has changed and border is accumulated, flush it. - if(border_colour_ != palette_[0] && duration_) { - flush_border(); - } - border_colour_ = palette_[0]; - // If the shifter is empty, accumulate in duration_ a promise to draw border later. if(!output_shifter_) { if(pixel_pointer_) { @@ -586,10 +582,17 @@ void Video::VideoStream::generate(int duration, OutputMode mode, bool is_termina } } +void Video::VideoStream::will_change_border_colour() { + // Flush the accumulated border if it'd be adversely affected. + if(duration_ && output_mode_ == OutputMode::Pixels) { + flush_border(); + } +} + void Video::VideoStream::flush_border() { // Output colour 0 for the entirety of duration_ (or black, if this is 1bpp mode). uint16_t *const colour_pointer = reinterpret_cast(crt_.begin_data(1)); - if(colour_pointer) *colour_pointer = (bpp_ != OutputBpp::One) ? border_colour_ : 0; + if(colour_pointer) *colour_pointer = (bpp_ != OutputBpp::One) ? palette_[0] : 0; crt_.output_level(duration_); duration_ = 0; diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index f6345821d..ef4c3c19d 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -177,7 +177,6 @@ class Video { public: VideoStream(Outputs::CRT::CRT &crt, uint16_t *palette) : crt_(crt), palette_(palette) {} - enum class OutputMode { Sync, Blank, ColourBurst, Pixels, }; @@ -185,8 +184,14 @@ class Video { /// Sets the current data format for the shifter. Changes in output BPP flush the shifter. void set_bpp(OutputBpp bpp); + /// Outputs signal of type @c mode for @c duration. void output(int duration, OutputMode mode); + /// Warns the video stream that the border colour, included in the palette that it holds a pointer to, + /// will change momentarily. This should be called after the relevant @c output() updates, and + /// is used to help elide border-regio output. + void will_change_border_colour(); + /// Loads 64 bits into the Shifter. The shifter shifts continuously. If you also declare /// a pixels region then whatever is being shifted will reach the display, in a form that /// depends on the current output BPP. @@ -208,7 +213,6 @@ class Video { // Internal state that is a function of output intent. int duration_ = 0; OutputMode output_mode_ = OutputMode::Sync; - uint16_t border_colour_ = 0; OutputBpp bpp_ = OutputBpp::Four; union { uint64_t output_shifter_; From 93f6964d8a6dd213e9d8ef290eaf2c5c6a087cdf Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 28 Dec 2019 22:50:34 -0500 Subject: [PATCH 07/18] Introduces some preliminary line length unit tests. Thereby fixes one potential issue with load_ toggling. --- Machines/Atari/ST/Video.cpp | 11 +- .../xcschemes/Clock Signal.xcscheme | 6 +- .../Clock SignalTests/AtariSTVideoTests.mm | 235 ++++++++++++++++-- 3 files changed, 227 insertions(+), 25 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index e0f01a5df..1285a2ee4 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -197,10 +197,6 @@ void Video::advance(HalfCycles duration) { } } - // TODO: if I'm asserting that sync and blank override the shifter (but, presumably, - // the shifter keeps shifting), then output_sync and output_blank need to have an effect - // inside the shifter on the temporary register values. - if(horizontal_.sync || vertical_.sync) { video_stream_.output(run_length, VideoStream::OutputMode::Sync); } else if(horizontal_.blank || vertical_.blank) { @@ -295,14 +291,17 @@ void Video::advance(HalfCycles duration) { x_ += run_length; integer_duration -= run_length; - // Check horizontal events. + // Check horizontal events; the first six are guaranteed to occur separately. if(horizontal_timings.reset_blank == x_) horizontal_.blank = false; else if(horizontal_timings.set_blank == x_) horizontal_.blank = true; else if(horizontal_timings.reset_enable == x_) horizontal_.enable = false; else if(horizontal_timings.set_enable == x_) horizontal_.enable = true; else if(line_length_ - hsync_start == x_) { horizontal_.sync = true; horizontal_.enable = false; } else if(line_length_ - hsync_end == x_) horizontal_.sync = false; - else if(next_load_toggle_ == x_) { + + // next_load_toggle_ is less predictable; test separately because it may coincide + // with one of the above tests. + if(next_load_toggle_ == x_) { next_load_toggle_ = -1; load_ ^= true; load_base_ = x_; diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme index 7b1a6914b..1465a4f62 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -23,9 +23,9 @@ diff --git a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm index 078c47718..305b732ab 100644 --- a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm +++ b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm @@ -8,30 +8,38 @@ #import +#include + #include "../../../Machines/Atari/ST/Video.hpp" @interface AtariSTVideoTests : XCTestCase @end -@implementation AtariSTVideoTests +@implementation AtariSTVideoTests { + std::unique_ptr _video; + uint16_t _ram[256*1024]; +} - (void)setUp { [super setUp]; + + // Establish an instance of video. + _video = std::make_unique(); + _video->set_ram(_ram, sizeof(_ram)); } - (void)tearDown { [super tearDown]; + + // Release the video instance. + _video = nullptr; } +/// Tests that no events occur outside of the sequence points the video predicts. - (void)testSequencePoints { - // Establish an instance of video. - Atari::ST::Video video; - uint16_t ram[256*1024]; - video.set_ram(ram, sizeof(ram)); - // Set 4bpp, 50Hz. - video.write(0x05, 0x0200); - video.write(0x30, 0x0000); + _video->write(0x05, 0x0200); + _video->write(0x30, 0x0000); // Run for [more than] a whole frame making sure that no observeable outputs // change at any time other than a sequence point. @@ -43,18 +51,213 @@ const bool is_transition_point = next_event == HalfCycles(0); if(is_transition_point) { - display_enable = video.display_enabled(); - vsync = video.vsync(); - hsync = video.hsync(); - next_event = video.get_next_sequence_point(); + display_enable = _video->display_enabled(); + vsync = _video->vsync(); + hsync = _video->hsync(); + next_event = _video->get_next_sequence_point(); } else { - NSAssert(display_enable == video.display_enabled(), @"Unannounced change in display enabled at cycle %zu [%d before next sequence point]", c, next_event.as()); - NSAssert(vsync == video.vsync(), @"Unannounced change in vsync at cycle %zu [%d before next sequence point]", c, next_event.as()); - NSAssert(hsync == video.hsync(), @"Unannounced change in hsync at cycle %zu [%d before next sequence point]", c, next_event.as()); + NSAssert(display_enable == _video->display_enabled(), @"Unannounced change in display enabled at cycle %zu [%d before next sequence point]", c, next_event.as()); + NSAssert(vsync == _video->vsync(), @"Unannounced change in vsync at cycle %zu [%d before next sequence point]", c, next_event.as()); + NSAssert(hsync == _video->hsync(), @"Unannounced change in hsync at cycle %zu [%d before next sequence point]", c, next_event.as()); } - video.run_for(HalfCycles(2)); + _video->run_for(HalfCycles(2)); next_event -= HalfCycles(2); } } +- (void)runVideoForCycles:(int)cycles { + while(cycles--) { + _video->run_for(Cycles(1)); + } +} + +- (void)syncToStartOfLine { + // Run until the visible fetch address changes, to get to the start of the pixel zone. + const uint32_t original_address = [self currentVideoAddress]; + while(original_address == [self currentVideoAddress]) { + _video->run_for(Cycles(1)); + } + + // Run until start of hsync. + while(!_video->hsync()) { + _video->run_for(Cycles(1)); + } + + // Run until end of hsync. + while(_video->hsync()) { + _video->run_for(Cycles(1)); + } + + // Run 12 cycles further. + [self runVideoForCycles:8]; +} + +- (void)setFrequency:(int)frequency { + switch(frequency) { + default: + case 50: _video->write(0x05, 0x200); _video->write(0x30, 0x000); break; + case 60: _video->write(0x05, 0x000); _video->write(0x30, 0x000); break; + case 72: _video->write(0x30, 0x200); break; + } +} + +- (uint32_t)currentVideoAddress { + return + (_video->read(0x04) & 0xff) | + ((_video->read(0x03) & 0xff) << 8) | + ((_video->read(0x02) & 0xff) << 16); +} + +struct RunLength { + int frequency; + int length; +}; + +- (void)testSequence:(const RunLength *)sequence targetLength:(int)duration { + [self syncToStartOfLine]; + + const uint32_t start_address = [self currentVideoAddress]; + + while(sequence->frequency != -1) { + [self setFrequency:sequence->frequency]; + [self runVideoForCycles:sequence->length]; + ++sequence; + } + const uint32_t final_address = [self currentVideoAddress]; + + XCTAssertEqual(final_address - start_address, duration); +} + +- (void)testLineLength54 { + // Run as though a regular 50Hz line at least until cycle 52; + // then switch to 72 Hz by 164, and allow the line to finish. + const RunLength test[] = { + {50, 60}, + {72, 452}, + {-1} + }; + [self testSequence:test targetLength:54]; +} + +- (void)testLineLength56 { + // Run as though a regular 60Hz line at least until cycle 52; + // then switch to 72 Hz by 164, and allow the line to finish. + const RunLength test[] = { + {60, 60}, + {72, 452}, + {-1} + }; + [self testSequence:test targetLength:56]; +} + +- (void)testLineLength80 { + // Run a standard 72Hz line. + const RunLength test[] = { + {72, 224}, + {-1} + }; + [self testSequence:test targetLength:80]; +} + +- (void)testLineLengthLong80 { + // Run a 72Hz line with a switch through 50Hz to extend the length to 512 cycles. + const RunLength test[] = { + {72, 50}, + {50, 20}, + {72, 442}, + {-1} + }; + [self testSequence:test targetLength:80]; +} + +- (void)testLineLength158 { + // Transition from 50Hz to 60Hz mid-line. + const RunLength test[] = { + {50, 60}, + {60, 458}, + {-1} + }; + [self testSequence:test targetLength:158]; +} + +- (void)testLineLength160_60Hz { + const RunLength test[] = { + {60, 508}, + {-1} + }; + [self testSequence:test targetLength:160]; +} + +- (void)testLineLength160_50Hz { + const RunLength test[] = { + {50, 512}, + {-1} + }; + [self testSequence:test targetLength:160]; +} + +- (void)testLineLength162 { + // Transition from 60Hz to 50Hz mid-line. + const RunLength test[] = { + {60, 54}, + {50, 458}, + {-1} + }; + [self testSequence:test targetLength:162]; +} + +- (void)testLineLength184 { + // Start off in 72Hz, switch to 60 during pixels. + const RunLength test[] = { + {72, 8}, + {60, 500}, + {-1} + }; + [self testSequence:test targetLength:184]; +} + +- (void)testLineLength186 { + // Start off in 72Hz, switch to 50 during pixels. + const RunLength test[] = { + {72, 8}, + {50, 504}, + {-1} + }; + [self testSequence:test targetLength:186]; +} + +- (void)testLineLength204 { + // Start in 50Hz, avoid DE disable. + const RunLength test[] = { + {50, 374}, + {60, 138}, + {-1} + }; + [self testSequence:test targetLength:204]; +} + +- (void)testLineLength206 { + // Start in 60Hz, get a 50Hz line length, avoid DE disable. + const RunLength test[] = { + {60, 53}, + {50, 3}, // To 56. + {60, 314}, // 370. + {50, 4}, // 374. + {60, 138}, // 512. + {-1} + }; + [self testSequence:test targetLength:206]; +} + +- (void)testLineLength230 { + // Start in 72Hz, avoid DE disable. + const RunLength test[] = { + {72, 8}, + {50, 366}, + {60, 138}, + {-1} + }; + [self testSequence:test targetLength:230]; +} + @end From 214b6a254ad722d7d63cd4a3dcc5035935aa5d02 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Dec 2019 17:37:09 -0500 Subject: [PATCH 08/18] Adds a delay on visibility of the hsync signal, and a test on address reload. --- Machines/Atari/ST/Video.cpp | 55 +++++---- Machines/Atari/ST/Video.hpp | 6 +- .../Clock SignalTests/AtariSTVideoTests.mm | 114 +++++++++++++----- 3 files changed, 122 insertions(+), 53 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 1285a2ee4..162b1226e 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -95,13 +95,17 @@ struct Checker { #endif const int de_delay_period = CYCLE(28); // Number of half cycles after DE that observed DE changes. -const int vsync_x_position = CYCLE(54); // Horizontal cycle on which vertical sync changes happen. -const int address_reload_x_position = CYCLE(54); // Horizontal cycle on which the base address is reloaded. +const int vsync_x_position = CYCLE(56); // Horizontal cycle on which vertical sync changes happen. + const int hsync_start = CYCLE(48); // Cycles before end of line when hsync starts. const int hsync_end = CYCLE(8); // Cycles before end of line when hsync ends. +const int hsync_delay_period = hsync_end; // Signal hsync at the end of the line. // "VSYNC starts 104 cycles after the start of the previous line's HSYNC, so that's 4 cycles before DE would be activated. "; -// hsync is at -50, so that's +54, or thereabouts. +// that's an inconsistent statement since it would imply VSYNC at +54, which is 2 cycles before DE in 60Hz mode and 6 before +// in 50Hz mode. I've gone with 56, to be four cycles ahead of DE in 50Hz mode. +// +// TODO: some sort of latency on vsync and hsync visibility, I would imagine? } @@ -178,6 +182,7 @@ void Video::advance(HalfCycles duration) { // Determine current output mode and number of cycles to output for. const int run_length = std::min(integer_duration, next_event - x_); const bool display_enable = vertical_.enable && horizontal_.enable; + const bool hsync = horizontal_.sync; // Ensure proper fetching irrespective of the output. if(load_) { @@ -243,23 +248,6 @@ void Video::advance(HalfCycles duration) { } } - // Check for address reload; this timing is _highly_ speculative on my part. I originally had it at frame end, - // then Enchanted Woods seemed to be doing its things three lines too late, so I moved it forward a little. - // Which didn't solve the problem, so I guess that title's logic isn't triggered on address reload, but it makes - // sense to keep it out separate and I've no better intel about its actual positioning. - if(y_ == vertical_timings.height - 3 && x_ <= address_reload_x_position && (x_ + run_length) > address_reload_x_position) { - current_address_ = base_address_ >> 1; - reset_fifo(); // TODO: remove this, probably, once otherwise stable? - - // Consider a shout out to the range observer. - if(previous_base_address_ != base_address_) { - previous_base_address_ = base_address_; - if(range_observer_) { - range_observer_->video_did_change_access_range(this); - } - } - } - // Check for whether line length should have been latched during this run. if(x_ <= CYCLE(54) && (x_ + run_length) > CYCLE(54)) line_length_ = horizontal_timings.length; @@ -278,11 +266,11 @@ void Video::advance(HalfCycles duration) { next_vertical_.enable = true; } else if(y_ == vertical_timings.reset_enable) { next_vertical_.enable = false; - } else if(next_y_ == vertical_timings.height - 1) { + } else if(next_y_ == vertical_timings.height - 2) { next_vertical_.sync_schedule = VerticalState::SyncSchedule::Begin; } else if(next_y_ == vertical_timings.height) { next_y_ = 0; - } else if(next_y_ == 2) { + } else if(y_ == 0) { next_vertical_.sync_schedule = VerticalState::SyncSchedule::End; } } @@ -311,6 +299,8 @@ void Video::advance(HalfCycles duration) { if(vertical_.sync_schedule != VerticalState::SyncSchedule::None && x_ == vsync_x_position) { vertical_.sync = vertical_.sync_schedule == VerticalState::SyncSchedule::Begin; vertical_.enable &= !vertical_.sync; + + reset_fifo(); // TODO: remove this, probably, once otherwise stable? } // Check whether the terminating event was end-of-line; if so then advance @@ -321,6 +311,20 @@ void Video::advance(HalfCycles duration) { y_ = next_y_; } + // The address is reloaded during the entire period of vertical sync. + // Cf. http://www.atari-forum.com/viewtopic.php?t=31954&start=50#p324730 + if(vertical_.sync) { + current_address_ = base_address_ >> 1; + + // Consider a shout out to the range observer. + if(previous_base_address_ != base_address_) { + previous_base_address_ = base_address_; + if(range_observer_) { + range_observer_->video_did_change_access_range(this); + } + } + } + // Chuck any deferred output changes into the queue. const bool next_display_enable = vertical_.enable && horizontal_.enable; if(display_enable != next_display_enable) { @@ -330,6 +334,11 @@ void Video::advance(HalfCycles duration) { // Schedule change in inwardly-visible effect. next_load_toggle_ = x_ + 8; // 4 cycles = 8 half-cycles } + + if(horizontal_.sync != hsync) { + // Schedule change in outwardly-visible hsync line. + add_event(hsync_delay_period - integer_duration, horizontal_.sync ? Event::Type::SetHsync : Event::Type::ResetHsync); + } } } @@ -351,7 +360,7 @@ void Video::reset_fifo() { } bool Video::hsync() { - return horizontal_.sync; + return public_state_.hsync; } bool Video::vsync() { diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index ef4c3c19d..b605d3c8c 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -227,12 +227,14 @@ class Video { /// Contains copies of the various observeable fields, after the relevant propagation delay. struct PublicState { bool display_enable = false; + bool hsync = false; } public_state_; struct Event { int delay; enum class Type { - SetDisplayEnable, ResetDisplayEnable + SetDisplayEnable, ResetDisplayEnable, + SetHsync, ResetHsync, } type; Event(Type type, int delay) : delay(delay), type(type) {} @@ -246,6 +248,8 @@ class Video { default: case Type::SetDisplayEnable: state.display_enable = true; break; case Type::ResetDisplayEnable: state.display_enable = false; break; + case Type::SetHsync: state.hsync = true; break; + case Type::ResetHsync: state.hsync = false; break; } } }; diff --git a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm index 305b732ab..70a4a6b45 100644 --- a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm +++ b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm @@ -20,6 +20,8 @@ uint16_t _ram[256*1024]; } +// MARK: - Setup and tear down. + - (void)setUp { [super setUp]; @@ -35,35 +37,7 @@ _video = nullptr; } -/// Tests that no events occur outside of the sequence points the video predicts. -- (void)testSequencePoints { - // Set 4bpp, 50Hz. - _video->write(0x05, 0x0200); - _video->write(0x30, 0x0000); - - // Run for [more than] a whole frame making sure that no observeable outputs - // change at any time other than a sequence point. - HalfCycles next_event; - bool display_enable = false; - bool vsync = false; - bool hsync = false; - for(size_t c = 0; c < 10 * 1000 * 1000; ++c) { - const bool is_transition_point = next_event == HalfCycles(0); - - if(is_transition_point) { - display_enable = _video->display_enabled(); - vsync = _video->vsync(); - hsync = _video->hsync(); - next_event = _video->get_next_sequence_point(); - } else { - NSAssert(display_enable == _video->display_enabled(), @"Unannounced change in display enabled at cycle %zu [%d before next sequence point]", c, next_event.as()); - NSAssert(vsync == _video->vsync(), @"Unannounced change in vsync at cycle %zu [%d before next sequence point]", c, next_event.as()); - NSAssert(hsync == _video->hsync(), @"Unannounced change in hsync at cycle %zu [%d before next sequence point]", c, next_event.as()); - } - _video->run_for(HalfCycles(2)); - next_event -= HalfCycles(2); - } -} +// MARK: - Helpers - (void)runVideoForCycles:(int)cycles { while(cycles--) { @@ -108,6 +82,45 @@ ((_video->read(0x02) & 0xff) << 16); } +- (void)setVideoBaseAddress:(uint32_t)baseAddress { + _video->write(0x00, baseAddress >> 16); + _video->write(0x01, baseAddress >> 8); +} + +// MARK: - Sequence Point Prediction Tests + +/// Tests that no events occur outside of the sequence points the video predicts. +- (void)testSequencePoints { + // Set 4bpp, 50Hz. + _video->write(0x05, 0x0200); + _video->write(0x30, 0x0000); + + // Run for [more than] a whole frame making sure that no observeable outputs + // change at any time other than a sequence point. + HalfCycles next_event; + bool display_enable = false; + bool vsync = false; + bool hsync = false; + for(size_t c = 0; c < 10 * 1000 * 1000; ++c) { + const bool is_transition_point = next_event == HalfCycles(0); + + if(is_transition_point) { + display_enable = _video->display_enabled(); + vsync = _video->vsync(); + hsync = _video->hsync(); + next_event = _video->get_next_sequence_point(); + } else { + NSAssert(display_enable == _video->display_enabled(), @"Unannounced change in display enabled at cycle %zu [%d before next sequence point]", c, next_event.as()); + NSAssert(vsync == _video->vsync(), @"Unannounced change in vsync at cycle %zu [%d before next sequence point]", c, next_event.as()); + NSAssert(hsync == _video->hsync(), @"Unannounced change in hsync at cycle %zu [%d before next sequence point]", c, next_event.as()); + } + _video->run_for(HalfCycles(2)); + next_event -= HalfCycles(2); + } +} + +// MARK: - Sync Line Length Tests + struct RunLength { int frequency; int length; @@ -260,4 +273,47 @@ struct RunLength { [self testSequence:test targetLength:230]; } +// MARK: - Address Reload Timing tests + +/// Tests that the current video address is reloaded constantly throughout hsync +- (void)testHsyncReload { + // Set an initial video address of 0. + [self setVideoBaseAddress:0]; + + // Find next area of non-vsync. + while(_video->vsync()) { + _video->run_for(Cycles(1)); + } + + // Set a different base video address. + [self setVideoBaseAddress:0x800000]; + + // Find next area of vsync, checking that the address isn't + // reloaded before then. + while(!_video->vsync()) { + XCTAssertNotEqual([self currentVideoAddress], 0x800000); + _video->run_for(Cycles(1)); + } + + // Vsync has now started, test that video address has been set. + XCTAssertEqual([self currentVideoAddress], 0x800000); + + // Run a few cycles, set a different video base address, + // confirm that has been set. + [self runVideoForCycles:200]; + XCTAssertEqual([self currentVideoAddress], 0x800000); + [self setVideoBaseAddress:0xc00000]; + [self runVideoForCycles:1]; + XCTAssertEqual([self currentVideoAddress], 0xc00000); + + // Find end of vertical sync, set a different base address, + // check that it doesn't become current. + while(_video->vsync()) { + _video->run_for(Cycles(1)); + } + [self setVideoBaseAddress:0]; + [self runVideoForCycles:1]; + XCTAssertNotEqual([self currentVideoAddress], 0); +} + @end From 5361ee252611cf11dd034763d4a5bca5dfea47ea Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Dec 2019 17:48:43 -0500 Subject: [PATCH 09/18] Adds specific Union Demo test. --- .../Mac/Clock SignalTests/AtariSTVideoTests.mm | 18 +++++++++++++++--- 1 file changed, 15 insertions(+), 3 deletions(-) diff --git a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm index 70a4a6b45..62a632cb5 100644 --- a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm +++ b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm @@ -61,9 +61,6 @@ while(_video->hsync()) { _video->run_for(Cycles(1)); } - - // Run 12 cycles further. - [self runVideoForCycles:8]; } - (void)setFrequency:(int)frequency { @@ -316,4 +313,19 @@ struct RunLength { XCTAssertNotEqual([self currentVideoAddress], 0); } +// MARK: - Tests Correlating To Exact Pieces of Software + +- (void)testUnionDemoScroller { + const RunLength test[] = { + {72, 8}, + {50, 365}, + {60, 8}, + {50, 59}, + {72, 12}, + {50, 60}, + {-1} + }; + [self testSequence:test targetLength:230]; +} + @end From 47068ee0814a70d30c63828ad93c635ba285ce5c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Dec 2019 17:51:50 -0500 Subject: [PATCH 10/18] Ensures visible hsync end generates a sequence point. --- Machines/Atari/ST/Video.cpp | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 162b1226e..cfd6fcf3c 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -412,9 +412,15 @@ HalfCycles Video::get_next_sequence_point() { event_time = std::min(event_time, vsync_x_position); } - // Test for beginning and end of horizontal sync. - if(x_ < line_length_ - hsync_start) event_time = std::min(line_length_ - hsync_start, event_time); - else if(x_ < line_length_ - hsync_end) event_time = std::min(line_length_ - hsync_end, event_time); + // Test for beginning and end of horizontal sync, and the times when those will actually be communicated. + if(x_ < line_length_ - hsync_start) { + event_time = std::min(line_length_ - hsync_start, event_time); + } else if(x_ < line_length_ - hsync_start + hsync_delay_period) { + event_time = std::min(line_length_ - hsync_start + hsync_delay_period, event_time); + } else if(x_ < line_length_ - hsync_end) { + event_time = std::min(line_length_ - hsync_end, event_time); + } + /* Assumed: hsync end will become visible at end of line. */ // It wasn't any of those, so as a temporary expedient, just supply end of line. return HalfCycles(event_time - x_); From 8ce26e7182146ae58bd91c1c126b5e38fb0b00db Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Dec 2019 19:03:08 -0500 Subject: [PATCH 11/18] Adds a delay on vsync visibility (i.e. as to generating an interrupt). --- Machines/Atari/ST/Video.cpp | 14 ++++++++++---- Machines/Atari/ST/Video.hpp | 4 ++++ 2 files changed, 14 insertions(+), 4 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index cfd6fcf3c..fd0e69d8c 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -99,13 +99,13 @@ const int vsync_x_position = CYCLE(56); // Horizontal cycle on which vertical s const int hsync_start = CYCLE(48); // Cycles before end of line when hsync starts. const int hsync_end = CYCLE(8); // Cycles before end of line when hsync ends. -const int hsync_delay_period = hsync_end; // Signal hsync at the end of the line. + +const int hsync_delay_period = hsync_end; // Signal hsync at the end of the line. +const int vsync_delay_period = hsync_delay_period; // Signal vsync with the same delay as hsync. // "VSYNC starts 104 cycles after the start of the previous line's HSYNC, so that's 4 cycles before DE would be activated. "; // that's an inconsistent statement since it would imply VSYNC at +54, which is 2 cycles before DE in 60Hz mode and 6 before // in 50Hz mode. I've gone with 56, to be four cycles ahead of DE in 50Hz mode. -// -// TODO: some sort of latency on vsync and hsync visibility, I would imagine? } @@ -183,6 +183,7 @@ void Video::advance(HalfCycles duration) { const int run_length = std::min(integer_duration, next_event - x_); const bool display_enable = vertical_.enable && horizontal_.enable; const bool hsync = horizontal_.sync; + const bool vsync = vertical_.sync; // Ensure proper fetching irrespective of the output. if(load_) { @@ -339,6 +340,11 @@ void Video::advance(HalfCycles duration) { // Schedule change in outwardly-visible hsync line. add_event(hsync_delay_period - integer_duration, horizontal_.sync ? Event::Type::SetHsync : Event::Type::ResetHsync); } + + if(vertical_.sync != vsync) { + // Schedule change in outwardly-visible hsync line. + add_event(vsync_delay_period - integer_duration, horizontal_.sync ? Event::Type::SetVsync : Event::Type::ResetVsync); + } } } @@ -364,7 +370,7 @@ bool Video::hsync() { } bool Video::vsync() { - return vertical_.sync; + return public_state_.vsync; } bool Video::display_enabled() { diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index b605d3c8c..371d0a4d7 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -228,6 +228,7 @@ class Video { struct PublicState { bool display_enable = false; bool hsync = false; + bool vsync = false; } public_state_; struct Event { @@ -235,6 +236,7 @@ class Video { enum class Type { SetDisplayEnable, ResetDisplayEnable, SetHsync, ResetHsync, + SetVsync, ResetVsync, } type; Event(Type type, int delay) : delay(delay), type(type) {} @@ -250,6 +252,8 @@ class Video { case Type::ResetDisplayEnable: state.display_enable = false; break; case Type::SetHsync: state.hsync = true; break; case Type::ResetHsync: state.hsync = false; break; + case Type::SetVsync: state.vsync = true; break; + case Type::ResetVsync: state.vsync = false; break; } } }; From b33218c61e2501d48b07cbb32fd89cbf38b801ff Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Dec 2019 20:55:34 -0500 Subject: [PATCH 12/18] Fixes reload test, which really needs to sense the CRT-headed vsync output. i.e. not the one heading back to the CPU. --- Machines/Atari/ST/Video.hpp | 5 +++++ .../Clock SignalTests/AtariSTVideoTests.mm | 19 ++++++++++++++----- 2 files changed, 19 insertions(+), 5 deletions(-) diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index 371d0a4d7..ef61a5609 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -15,6 +15,9 @@ #include +// Testing hook; not for any other user. +class VideoTester; + namespace Atari { namespace ST { @@ -278,6 +281,8 @@ class Video { } pending_events_.emplace(insertion_point, type, delay); } + + friend class ::VideoTester; }; } diff --git a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm index 62a632cb5..f2419e096 100644 --- a/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm +++ b/OSBindings/Mac/Clock SignalTests/AtariSTVideoTests.mm @@ -12,6 +12,14 @@ #include "../../../Machines/Atari/ST/Video.hpp" +// Implement Atari::ST::Video's friend class, to expose some +// internal state. +struct VideoTester { + static bool vsync(Atari::ST::Video &video) { + return video.vertical_.sync; + } +}; + @interface AtariSTVideoTests : XCTestCase @end @@ -272,13 +280,14 @@ struct RunLength { // MARK: - Address Reload Timing tests -/// Tests that the current video address is reloaded constantly throughout hsync -- (void)testHsyncReload { +/// Tests that the current video address is reloaded constantly throughout vsync (subject to caveat: observed . +- (void)testVsyncReload { + // Set an initial video address of 0. [self setVideoBaseAddress:0]; // Find next area of non-vsync. - while(_video->vsync()) { + while(VideoTester::vsync(*_video)) { _video->run_for(Cycles(1)); } @@ -287,7 +296,7 @@ struct RunLength { // Find next area of vsync, checking that the address isn't // reloaded before then. - while(!_video->vsync()) { + while(!VideoTester::vsync(*_video)) { XCTAssertNotEqual([self currentVideoAddress], 0x800000); _video->run_for(Cycles(1)); } @@ -305,7 +314,7 @@ struct RunLength { // Find end of vertical sync, set a different base address, // check that it doesn't become current. - while(_video->vsync()) { + while(VideoTester::vsync(*_video)) { _video->run_for(Cycles(1)); } [self setVideoBaseAddress:0]; From c8fe66092bc5741a9b47a41172597c778aac2322 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Dec 2019 21:42:41 -0500 Subject: [PATCH 13/18] Attempts to correct insertion logic (and mostly bypasses it). --- Machines/Atari/ST/Video.hpp | 25 +++++++++++++++---------- 1 file changed, 15 insertions(+), 10 deletions(-) diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index ef61a5609..4f9f06dd5 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -269,17 +269,22 @@ class Video { return; } - // Otherwise enqueue, having subtracted the delay for any preceding events, - // and subtracting from the subsequent, if any. - auto insertion_point = pending_events_.begin(); - while(insertion_point != pending_events_.end() && insertion_point->delay > delay) { - delay -= insertion_point->delay; - ++insertion_point; + if(!pending_events_.empty()) { + // Otherwise enqueue, having subtracted the delay for any preceding events, + // and subtracting from the subsequent, if any. + auto insertion_point = pending_events_.begin(); + while(insertion_point != pending_events_.end() && insertion_point->delay < delay) { + delay -= insertion_point->delay; + ++insertion_point; + } + if(insertion_point != pending_events_.end()) { + insertion_point->delay -= delay; + } + + pending_events_.emplace(insertion_point, type, delay); + } else { + pending_events_.emplace_back(type, delay); } - if(insertion_point != pending_events_.end()) { - insertion_point->delay -= delay; - } - pending_events_.emplace(insertion_point, type, delay); } friend class ::VideoTester; From 6449403f6a9e51712a627fe4de5ba0ffbb6416f8 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Dec 2019 21:53:45 -0500 Subject: [PATCH 14/18] Corrects pending_events_ test for sequence points. Simplifies around as possible. --- Machines/Atari/ST/Video.cpp | 15 ++++++--------- .../xcshareddata/xcschemes/Clock Signal.xcscheme | 2 +- 2 files changed, 7 insertions(+), 10 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index fd0e69d8c..9da6b1afd 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -400,7 +400,7 @@ HalfCycles Video::get_next_sequence_point() { // If any events are pending, give the first of those the chance to be next. if(!pending_events_.empty()) { - event_time = std::min(event_time, x_ + event_time); + event_time = std::min(event_time, x_ + pending_events_.front().delay); } // If this is a vertically-enabled line, check for the display enable boundaries, + the standard delay. @@ -418,17 +418,14 @@ HalfCycles Video::get_next_sequence_point() { event_time = std::min(event_time, vsync_x_position); } - // Test for beginning and end of horizontal sync, and the times when those will actually be communicated. - if(x_ < line_length_ - hsync_start) { - event_time = std::min(line_length_ - hsync_start, event_time); - } else if(x_ < line_length_ - hsync_start + hsync_delay_period) { + // Test for beginning and end of horizontal sync. + if(x_ < line_length_ - hsync_start + hsync_delay_period) { event_time = std::min(line_length_ - hsync_start + hsync_delay_period, event_time); - } else if(x_ < line_length_ - hsync_end) { - event_time = std::min(line_length_ - hsync_end, event_time); } - /* Assumed: hsync end will become visible at end of line. */ + /* Hereby assumed: hsync end will be communicated at end of line: */ + static_assert(hsync_end == hsync_delay_period); - // It wasn't any of those, so as a temporary expedient, just supply end of line. + // It wasn't any of those, just supply end of line. That's when the static_assert above assumes a visible hsync transition. return HalfCycles(event_time - x_); } diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme index 1465a4f62..47f9c7286 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -67,7 +67,7 @@ Date: Sun, 29 Dec 2019 22:03:36 -0500 Subject: [PATCH 15/18] Restores vsync active. --- Machines/Atari/ST/Video.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 9da6b1afd..73a90536c 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -343,7 +343,7 @@ void Video::advance(HalfCycles duration) { if(vertical_.sync != vsync) { // Schedule change in outwardly-visible hsync line. - add_event(vsync_delay_period - integer_duration, horizontal_.sync ? Event::Type::SetVsync : Event::Type::ResetVsync); + add_event(vsync_delay_period - integer_duration, vertical_.sync ? Event::Type::SetVsync : Event::Type::ResetVsync); } } } From 09513ec14cd13407e38962694e31ede14e3b922e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 30 Dec 2019 22:58:19 -0500 Subject: [PATCH 16/18] Gets explicit about constexpr expectations here. --- ClockReceiver/JustInTime.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ClockReceiver/JustInTime.hpp b/ClockReceiver/JustInTime.hpp index 122344dcf..cc83a0f90 100644 --- a/ClockReceiver/JustInTime.hpp +++ b/ClockReceiver/JustInTime.hpp @@ -29,7 +29,7 @@ template ()); } else { const auto duration = time_since_update_.template divide(LocalTimeScale(divider)); From 8e777c299ff312b35193a65ec959bfeaeee525e3 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 30 Dec 2019 23:00:55 -0500 Subject: [PATCH 17/18] Switches to latching video interrupts until acknowledged. Seems to fix Cisco Heat, at least. I have no idea whether I'm latching the correct thing, whether IACK should clear both or only one, etc. --- Machines/Atari/ST/AtariST.cpp | 24 +++++++++++++++++++++--- 1 file changed, 21 insertions(+), 3 deletions(-) diff --git a/Machines/Atari/ST/AtariST.cpp b/Machines/Atari/ST/AtariST.cpp index 52c66b41c..e29c45069 100644 --- a/Machines/Atari/ST/AtariST.cpp +++ b/Machines/Atari/ST/AtariST.cpp @@ -176,7 +176,10 @@ class ConcreteMachine: // An interrupt acknowledge, perhaps? if(cycle.operation & Microcycle::InterruptAcknowledge) { // Current implementation: everything other than 6 (i.e. the MFP is autovectored. - if((cycle.word_address()&7) != 6) { + const int interrupt_level = cycle.word_address()&7; + if(interrupt_level != 6) { + video_interrupts_pending_ &= ~interrupt_level; + update_interrupt_input(); mc68000_.set_is_peripheral_address(true); return HalfCycles(0); } else { @@ -552,12 +555,27 @@ class ConcreteMachine: update_interrupt_input(); } + int video_interrupts_pending_ = 0; + bool previous_hsync_ = false, previous_vsync_ = false; void update_interrupt_input() { + // Complete guess: set video interrupts pending if/when hsync of vsync + // go inactive. Reset upon IACK. + const bool hsync = video_.last_valid()->hsync(); + const bool vsync = video_.last_valid()->vsync(); + if(previous_hsync_ != hsync && previous_hsync_) { + video_interrupts_pending_ |= 2; + } + if(previous_vsync_ != vsync && previous_vsync_) { + video_interrupts_pending_ |= 4; + } + previous_vsync_ = vsync; + previous_hsync_ = hsync; + if(mfp_->get_interrupt_line()) { mc68000_.set_interrupt_level(6); - } else if(video_->vsync()) { + } else if(video_interrupts_pending_ & 4) { mc68000_.set_interrupt_level(4); - } else if(video_->hsync()) { + } else if(video_interrupts_pending_ & 2) { mc68000_.set_interrupt_level(2); } else { mc68000_.set_interrupt_level(0); From 0a12893d63f75030bf8ad60d777d207b04070104 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 30 Dec 2019 23:01:31 -0500 Subject: [PATCH 18/18] Shunts vsync back down to top of frame. It's guess after guess, basically. --- Machines/Atari/ST/Video.cpp | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 73a90536c..f2977fe75 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -267,11 +267,10 @@ void Video::advance(HalfCycles duration) { next_vertical_.enable = true; } else if(y_ == vertical_timings.reset_enable) { next_vertical_.enable = false; - } else if(next_y_ == vertical_timings.height - 2) { - next_vertical_.sync_schedule = VerticalState::SyncSchedule::Begin; } else if(next_y_ == vertical_timings.height) { + next_vertical_.sync_schedule = VerticalState::SyncSchedule::Begin; next_y_ = 0; - } else if(y_ == 0) { + } else if(next_y_ == 2) { next_vertical_.sync_schedule = VerticalState::SyncSchedule::End; } }