From 0ffded72a6c58760d1c45dfad0fe187eccded9df Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 28 Jan 2017 16:19:08 -0500 Subject: [PATCH 01/91] Created a placeholder class for a factored-out TIA. There's a bit more it'll need to do, like vending (or receiving) a CRT but this is the full hardware stuff, I think. --- Machines/Atari2600/TIA.cpp | 9 +++ Machines/Atari2600/TIA.hpp | 58 +++++++++++++++++++ .../Clock Signal.xcodeproj/project.pbxproj | 8 ++- 3 files changed, 74 insertions(+), 1 deletion(-) create mode 100644 Machines/Atari2600/TIA.cpp create mode 100644 Machines/Atari2600/TIA.hpp diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp new file mode 100644 index 000000000..bac602803 --- /dev/null +++ b/Machines/Atari2600/TIA.cpp @@ -0,0 +1,9 @@ +// +// TIA.cpp +// Clock Signal +// +// Created by Thomas Harte on 28/01/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#include "TIA.hpp" diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp new file mode 100644 index 000000000..24d3376ad --- /dev/null +++ b/Machines/Atari2600/TIA.hpp @@ -0,0 +1,58 @@ +// +// TIA.hpp +// Clock Signal +// +// Created by Thomas Harte on 28/01/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#ifndef TIA_hpp +#define TIA_hpp + +#include + +namespace Atari2600 { + +class TIA { + public: + void run_for_cycles(int number_of_cycles); + + void set_vsync(bool vsync); + void set_vblank(bool vblank); + void reset_horizontal_counter(); + int get_cycles_until_horizontal_blank(); + + void set_background_colour(uint8_t colour); + + void set_playfield(uint16_t offset, uint8_t value); + void set_playfield_control_and_ball_size(uint8_t value); + void set_playfield_ball_colour(uint8_t colour); + + void set_player_number_and_size(int player, uint8_t value); + void set_player_graphic(int player, uint8_t value); + void set_player_reflected(int player, bool reflected); + void set_player_delay(int player, bool delay); + void set_player_position(int player); + void set_player_motion(int player, uint8_t motion); + void set_player_missile_colour(int player, uint8_t colour); + + void set_missile_enable(int missile, bool enabled); + void set_missile_position(int missile); + void set_missile_position_to_player(int missile); + void set_missile_motion(int missile, uint8_t motion); + + void set_ball_enable(bool enabled); + void set_ball_delay(bool delay); + void set_ball_position(); + void set_ball_motion(uint8_t motion); + + void move(); + void clear_motion(); + + uint8_t get_collision_flags(int offset); + void clear_collision_flags(); +}; + +} + +#endif /* TIA_hpp */ diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 18332f71c..08c8373f8 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -396,6 +396,7 @@ 4BD5F1951D13528900631CD1 /* CSBestEffortUpdater.m in Sources */ = {isa = PBXBuildFile; fileRef = 4BD5F1941D13528900631CD1 /* CSBestEffortUpdater.m */; }; 4BD69F941D98760000243FE1 /* AcornADF.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BD69F921D98760000243FE1 /* AcornADF.cpp */; }; 4BE77A2E1D84ADFB00BC3827 /* File.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE77A2C1D84ADFB00BC3827 /* File.cpp */; }; + 4BE7C9181E3D397100A5496D /* TIA.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE7C9161E3D397100A5496D /* TIA.cpp */; }; 4BEA525E1DF33323007E74F2 /* Tape.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BEA525D1DF33323007E74F2 /* Tape.cpp */; }; 4BEA52631DF339D7007E74F2 /* Speaker.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BEA52611DF339D7007E74F2 /* Speaker.cpp */; }; 4BEA52661DF3472B007E74F2 /* Speaker.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BEA52641DF3472B007E74F2 /* Speaker.cpp */; }; @@ -923,6 +924,8 @@ 4BD69F931D98760000243FE1 /* AcornADF.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = AcornADF.hpp; sourceTree = ""; }; 4BE77A2C1D84ADFB00BC3827 /* File.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = File.cpp; path = ../../StaticAnalyser/Commodore/File.cpp; sourceTree = ""; }; 4BE77A2D1D84ADFB00BC3827 /* File.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = File.hpp; path = ../../StaticAnalyser/Commodore/File.hpp; sourceTree = ""; }; + 4BE7C9161E3D397100A5496D /* TIA.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = TIA.cpp; sourceTree = ""; }; + 4BE7C9171E3D397100A5496D /* TIA.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = TIA.hpp; sourceTree = ""; }; 4BEA525D1DF33323007E74F2 /* Tape.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = Tape.cpp; path = Electron/Tape.cpp; sourceTree = ""; }; 4BEA525F1DF333D8007E74F2 /* Tape.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; name = Tape.hpp; path = Electron/Tape.hpp; sourceTree = ""; }; 4BEA52601DF3343A007E74F2 /* Interrupts.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; name = Interrupts.hpp; path = Electron/Interrupts.hpp; sourceTree = ""; }; @@ -1094,10 +1097,12 @@ children = ( 4B2E2D971C3A06EC00138695 /* Atari2600.cpp */, 4BEA52641DF3472B007E74F2 /* Speaker.cpp */, + 4BE7C9161E3D397100A5496D /* TIA.cpp */, 4B2E2D991C3A06EC00138695 /* Atari2600Inputs.h */, 4B2E2D981C3A06EC00138695 /* Atari2600.hpp */, - 4BEA52651DF3472B007E74F2 /* Speaker.hpp */, 4BEA52671DF34909007E74F2 /* PIA.hpp */, + 4BEA52651DF3472B007E74F2 /* Speaker.hpp */, + 4BE7C9171E3D397100A5496D /* TIA.hpp */, ); path = Atari2600; sourceTree = ""; @@ -2434,6 +2439,7 @@ 4B643F3A1D77AD1900D431D6 /* CSStaticAnalyser.mm in Sources */, 4B4DC8281D2C2470003C5BF8 /* C1540.cpp in Sources */, 4B5A12571DD55862007A2231 /* Disassembler6502.cpp in Sources */, + 4BE7C9181E3D397100A5496D /* TIA.cpp in Sources */, 4B1E85751D170228001EF87D /* Typer.cpp in Sources */, 4BF829631D8F536B001BAE39 /* SSD.cpp in Sources */, 4B2E2D9D1C3A070400138695 /* Electron.cpp in Sources */, From a246530953ca0e0315ac387030d6a00c6888ff11 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 28 Jan 2017 21:46:40 -0500 Subject: [PATCH 02/91] Supposing the TIA were implemented, this is (more or less) what the Atari 2600 would now look like. --- Machines/Atari2600/Atari2600.cpp | 600 +++---------------------------- Machines/Atari2600/Atari2600.hpp | 111 +----- Machines/Atari2600/TIA.cpp | 49 +++ Machines/Atari2600/TIA.hpp | 14 +- 4 files changed, 121 insertions(+), 653 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index ea96ea83c..75db7269f 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -18,91 +18,25 @@ namespace { } Machine::Machine() : - horizontal_timer_(0), - last_output_state_duration_(0), - last_output_state_(OutputState::Sync), rom_(nullptr), + rom_pages_{nullptr, nullptr, nullptr, nullptr}, tia_input_value_{0xff, 0xff}, - upcoming_events_pointer_(0), - object_counter_pointer_(0), - state_by_time_(state_by_extend_time_[0]), cycles_since_speaker_update_(0), - is_pal_region_(false) + cycles_since_video_update_(0) { - memset(collisions_, 0xff, sizeof(collisions_)); - setup_reported_collisions(); - - for(int vbextend = 0; vbextend < 2; vbextend++) - { - for(int c = 0; c < 57; c++) - { - OutputState state; - - // determine which output state will be active in four cycles from now - switch(c) - { - case 0: case 1: case 2: case 3: state = OutputState::Blank; break; - case 4: case 5: case 6: case 7: state = OutputState::Sync; break; - case 8: case 9: case 10: case 11: state = OutputState::ColourBurst; break; - case 12: case 13: case 14: - case 15: case 16: state = OutputState::Blank; break; - - case 17: case 18: state = vbextend ? OutputState::Blank : OutputState::Pixel; break; - default: state = OutputState::Pixel; break; - } - - state_by_extend_time_[vbextend][c] = state; - } - } set_clock_rate(NTSC_clock_rate); } void Machine::setup_output(float aspect_ratio) { + tia_.reset(new TIA); speaker_.reset(new Speaker); - crt_.reset(new Outputs::CRT::CRT(228, 1, 263, Outputs::CRT::ColourSpace::YIQ, 228, 1, false, 1)); - crt_->set_output_device(Outputs::CRT::Television); - - // this is the NTSC phase offset function; see below for PAL - crt_->set_composite_sampling_function( - "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" - "{" - "uint c = texture(texID, coordinate).r;" - "uint y = c & 14u;" - "uint iPhase = (c >> 4);" - - "float phaseOffset = 6.283185308 * float(iPhase) / 13.0 + 5.074880441076923;" - "return mix(float(y) / 14.0, step(1, iPhase) * cos(phase + phaseOffset), amplitude);" - "}"); - speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); -} - -void Machine::switch_region() -{ - // the PAL function - crt_->set_composite_sampling_function( - "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" - "{" - "uint c = texture(texID, coordinate).r;" - "uint y = c & 14u;" - "uint iPhase = (c >> 4);" - - "uint direction = iPhase & 1u;" - "float phaseOffset = float(7u - direction) + (float(direction) - 0.5) * 2.0 * float(iPhase >> 1);" - "phaseOffset *= 6.283185308 / 12.0;" - "return mix(float(y) / 14.0, step(4, (iPhase + 2u) & 15u) * cos(phase + phaseOffset), amplitude);" - "}"); - - crt_->set_new_timing(228, 312, Outputs::CRT::ColourSpace::YUV, 228, 1, true); - - is_pal_region_ = true; - speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); - set_clock_rate(PAL_clock_rate); } void Machine::close_output() { - crt_ = nullptr; + tia_ = nullptr; + speaker_ = nullptr; } Machine::~Machine() @@ -111,303 +45,6 @@ Machine::~Machine() close_output(); } -void Machine::update_timers(int mask) -{ - unsigned int upcoming_pointer_plus_4 = (upcoming_events_pointer_ + 4)%number_of_upcoming_events; - - object_counter_pointer_ = (object_counter_pointer_ + 1)%number_of_recorded_counters; - ObjectCounter *oneClockAgo = object_counter_[(object_counter_pointer_ - 1 + number_of_recorded_counters)%number_of_recorded_counters]; - ObjectCounter *twoClocksAgo = object_counter_[(object_counter_pointer_ - 2 + number_of_recorded_counters)%number_of_recorded_counters]; - ObjectCounter *now = object_counter_[object_counter_pointer_]; - - // grab the background now, for application in four clocks - if(mask & (1 << 5) && !(horizontal_timer_&3)) - { - unsigned int offset = 4 + horizontal_timer_ - (horizontalTimerPeriod - 160); - upcoming_events_[upcoming_pointer_plus_4].updates |= Event::Action::Playfield; - upcoming_events_[upcoming_pointer_plus_4].playfield_pixel = playfield_[(offset >> 2)%40]; - } - - if(mask & (1 << 4)) - { - // the ball becomes visible whenever it hits zero, regardless of whether its status - // is the result of a counter rollover or a programmatic reset, and there's a four - // clock delay on that triggering the start signal - now[4].count = (oneClockAgo[4].count + 1)%160; - now[4].pixel = oneClockAgo[4].pixel + 1; - if(!now[4].count) now[4].pixel = 0; - } - else - { - now[4] = oneClockAgo[4]; - } - - // check for player and missle triggers - for(int c = 0; c < 4; c++) - { - if(mask & (1 << c)) - { - // update the count - now[c].count = (oneClockAgo[c].count + 1)%160; - - uint8_t repeatMask = player_and_missile_size_[c&1] & 7; - ObjectCounter *rollover; - ObjectCounter *equality; - - if(c < 2) - { - // update the pixel - now[c].broad_pixel = oneClockAgo[c].broad_pixel + 1; - switch(repeatMask) - { - default: now[c].pixel = oneClockAgo[c].pixel + 1; break; - case 5: now[c].pixel = oneClockAgo[c].pixel + (now[c].broad_pixel&1); break; - case 7: now[c].pixel = oneClockAgo[c].pixel + (((now[c].broad_pixel | (now[c].broad_pixel >> 1))^1)&1); break; - } - - // check for a rollover six clocks ago or equality five clocks ago - rollover = twoClocksAgo; - equality = oneClockAgo; - } - else - { - // update the pixel - now[c].pixel = oneClockAgo[c].pixel + 1; - - // check for a rollover five clocks ago or equality four clocks ago - rollover = oneClockAgo; - equality = now; - } - - if( - (rollover[c].count == 159) || - (has_second_copy_[c&1] && equality[c].count == 16) || - (has_third_copy_[c&1] && equality[c].count == 32) || - (has_fourth_copy_[c&1] && equality[c].count == 64) - ) - { - now[c].pixel = 0; - now[c].broad_pixel = 0; - } - } - else - { - now[c] = oneClockAgo[c]; - } - } -} - -uint8_t Machine::get_output_pixel() -{ - ObjectCounter *now = object_counter_[object_counter_pointer_]; - - // get the playfield pixel - unsigned int offset = horizontal_timer_ - (horizontalTimerPeriod - 160); - uint8_t playfieldColour = ((playfield_control_&6) == 2) ? player_colour_[offset / 80] : playfield_colour_; - - // ball pixel - uint8_t ballPixel = 0; - if(now[4].pixel < ball_size_) { - ballPixel = ball_graphics_enable_[ball_graphics_selector_]; - } - - // determine the player and missile pixels - uint8_t playerPixels[2] = { 0, 0 }; - uint8_t missilePixels[2] = { 0, 0 }; - for(int c = 0; c < 2; c++) - { - if(player_graphics_[c] && now[c].pixel < 8) { - playerPixels[c] = (player_graphics_[player_graphics_selector_[c]][c] >> (now[c].pixel ^ player_reflection_mask_[c])) & 1; - } - - if(!missile_graphics_reset_[c] && now[c+2].pixel < missile_size_[c]) { - missilePixels[c] = missile_graphics_enable_[c]; - } - } - - // accumulate collisions - int pixel_mask = playerPixels[0] | (playerPixels[1] << 1) | (missilePixels[0] << 2) | (missilePixels[1] << 3) | (ballPixel << 4) | (playfield_output_ << 5); - collisions_[0] |= reported_collisions_[pixel_mask][0]; - collisions_[1] |= reported_collisions_[pixel_mask][1]; - collisions_[2] |= reported_collisions_[pixel_mask][2]; - collisions_[3] |= reported_collisions_[pixel_mask][3]; - collisions_[4] |= reported_collisions_[pixel_mask][4]; - collisions_[5] |= reported_collisions_[pixel_mask][5]; - collisions_[6] |= reported_collisions_[pixel_mask][6]; - collisions_[7] |= reported_collisions_[pixel_mask][7]; - - // apply appropriate priority to pick a colour - uint8_t playfield_pixel = playfield_output_ | ballPixel; - uint8_t outputColour = playfield_pixel ? playfieldColour : background_colour_; - - if(!(playfield_control_&0x04) || !playfield_pixel) { - if(playerPixels[1] || missilePixels[1]) outputColour = player_colour_[1]; - if(playerPixels[0] || missilePixels[0]) outputColour = player_colour_[0]; - } - - // return colour - return outputColour; -} - -void Machine::setup_reported_collisions() -{ - for(int c = 0; c < 64; c++) - { - memset(reported_collisions_[c], 0, 8); - - int playerPixels[2] = { c&1, (c >> 1)&1 }; - int missilePixels[2] = { (c >> 2)&1, (c >> 3)&1 }; - int ballPixel = (c >> 4)&1; - int playfield_pixel = (c >> 5)&1; - - if(playerPixels[0] | playerPixels[1]) { - reported_collisions_[c][0] |= ((missilePixels[0] & playerPixels[1]) << 7) | ((missilePixels[0] & playerPixels[0]) << 6); - reported_collisions_[c][1] |= ((missilePixels[1] & playerPixels[0]) << 7) | ((missilePixels[1] & playerPixels[1]) << 6); - - reported_collisions_[c][2] |= ((playfield_pixel & playerPixels[0]) << 7) | ((ballPixel & playerPixels[0]) << 6); - reported_collisions_[c][3] |= ((playfield_pixel & playerPixels[1]) << 7) | ((ballPixel & playerPixels[1]) << 6); - - reported_collisions_[c][7] |= ((playerPixels[0] & playerPixels[1]) << 7); - } - - if(playfield_pixel | ballPixel) { - reported_collisions_[c][4] |= ((playfield_pixel & missilePixels[0]) << 7) | ((ballPixel & missilePixels[0]) << 6); - reported_collisions_[c][5] |= ((playfield_pixel & missilePixels[1]) << 7) | ((ballPixel & missilePixels[1]) << 6); - - reported_collisions_[c][6] |= ((playfield_pixel & ballPixel) << 7); - } - - if(missilePixels[0] & missilePixels[1]) - reported_collisions_[c][7] |= (1 << 6); - } -} - -void Machine::output_pixels(unsigned int count) -{ - while(count--) - { - if(upcoming_events_[upcoming_events_pointer_].updates) - { - // apply any queued changes and flush the record - if(upcoming_events_[upcoming_events_pointer_].updates & Event::Action::HMoveSetup) - { - // schedule an extended left border - state_by_time_ = state_by_extend_time_[1]; - - // clear any ongoing moves - if(hmove_flags_) - { - for(int c = 0; c < number_of_upcoming_events; c++) - { - upcoming_events_[c].updates &= ~(Event::Action::HMoveCompare | Event::Action::HMoveDecrement); - } - } - - // schedule new moves - hmove_flags_ = 0x1f; - hmove_counter_ = 15; - - // follow-through into a compare immediately - upcoming_events_[upcoming_events_pointer_].updates |= Event::Action::HMoveCompare; - } - - if(upcoming_events_[upcoming_events_pointer_].updates & Event::Action::HMoveCompare) - { - for(int c = 0; c < 5; c++) - { - if(((object_motions_[c] >> 4)^hmove_counter_) == 7) - { - hmove_flags_ &= ~(1 << c); - } - } - if(hmove_flags_) - { - if(hmove_counter_) hmove_counter_--; - upcoming_events_[(upcoming_events_pointer_+4)%number_of_upcoming_events].updates |= Event::Action::HMoveCompare; - upcoming_events_[(upcoming_events_pointer_+2)%number_of_upcoming_events].updates |= Event::Action::HMoveDecrement; - } - } - - if(upcoming_events_[upcoming_events_pointer_].updates & Event::Action::HMoveDecrement) - { - update_timers(hmove_flags_); - } - - if(upcoming_events_[upcoming_events_pointer_].updates & Event::Action::ResetCounter) - { - object_counter_[object_counter_pointer_][upcoming_events_[upcoming_events_pointer_].counter].count = 0; - } - - // zero out current update event - upcoming_events_[upcoming_events_pointer_].updates = 0; - } - - // progress to next event - upcoming_events_pointer_ = (upcoming_events_pointer_ + 1)%number_of_upcoming_events; - - // determine which output state is currently active - OutputState primary_state = state_by_time_[horizontal_timer_ >> 2]; - OutputState effective_state = primary_state; - - // update pixel timers - if(primary_state == OutputState::Pixel) update_timers(~0); - - // update the background chain - if(horizontal_timer_ >= 64 && horizontal_timer_ <= 160+64 && !(horizontal_timer_&3)) - { - playfield_output_ = next_playfield_output_; - next_playfield_output_ = playfield_[(horizontal_timer_ - 64) >> 2]; - } - - // if vsync is enabled, output the opposite of the automatic hsync output; - // also honour the vertical blank flag - if(vsync_enabled_) { - effective_state = (effective_state = OutputState::Sync) ? OutputState::Blank : OutputState::Sync; - } else if(vblank_enabled_ && effective_state == OutputState::Pixel) { - effective_state = OutputState::Blank; - } - - // decide what that means needs to be communicated to the CRT - last_output_state_duration_++; - if(effective_state != last_output_state_) { - switch(last_output_state_) { - case OutputState::Blank: crt_->output_blank(last_output_state_duration_); break; - case OutputState::Sync: crt_->output_sync(last_output_state_duration_); break; - case OutputState::ColourBurst: crt_->output_colour_burst(last_output_state_duration_, 96, 0); break; - case OutputState::Pixel: crt_->output_data(last_output_state_duration_, 1); break; - } - last_output_state_duration_ = 0; - last_output_state_ = effective_state; - - if(effective_state == OutputState::Pixel) { - output_buffer_ = crt_->allocate_write_area(160); - } else { - output_buffer_ = nullptr; - } - } - - // decide on a pixel colour if that's what's happening - if(effective_state == OutputState::Pixel) - { - uint8_t colour = get_output_pixel(); - if(output_buffer_) - { - *output_buffer_ = colour; - output_buffer_++; - } - } - - // advance horizontal timer, perform reset actions if desired - horizontal_timer_ = (horizontal_timer_ + 1) % horizontalTimerPeriod; - if(!horizontal_timer_) - { - // switch back to a normal length left border - state_by_time_ = state_by_extend_time_[0]; - set_ready_line(false); - } - } -} - unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uint16_t address, uint8_t *value) { uint8_t returnValue = 0xff; @@ -418,12 +55,11 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin // effect until the next read; therefore it isn't safe to assume that signalling ready immediately // skips to the end of the line. if(operation == CPU6502::BusOperation::Ready) { - unsigned int distance_to_end_of_ready = horizontalTimerPeriod - horizontal_timer_; - cycles_run_for = distance_to_end_of_ready; + cycles_run_for = (unsigned int)tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_); } - output_pixels(cycles_run_for); cycles_since_speaker_update_ += cycles_run_for; + cycles_since_video_update_ += cycles_run_for; if(operation != CPU6502::BusOperation::Ready) { @@ -475,7 +111,7 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin case 0x05: // missile 1 / playfield / ball collisions case 0x06: // ball / playfield collisions case 0x07: // player / player, missile / missile collisions - returnValue &= collisions_[decodedAddress]; + returnValue &= tia_->get_collision_flags(decodedAddress); break; case 0x08: @@ -493,194 +129,58 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin } else { const uint16_t decodedAddress = address & 0x3f; switch(decodedAddress) { - case 0x00: - vsync_enabled_ = !!(*value & 0x02); - break; - case 0x01: vblank_enabled_ = !!(*value & 0x02); break; + case 0x00: update_video(); tia_->set_vsync(!!(*value & 0x02)); break; + case 0x01: update_video(); tia_->set_vblank(!!(*value & 0x02)); break; case 0x02: - if(horizontal_timer_) set_ready_line(true); + set_ready_line(true); + // TODO: if(horizontal_timer_) break; - case 0x03: - // Reset is delayed by four cycles. - horizontal_timer_ = horizontalTimerPeriod - 4; - + case 0x03: update_video(); tia_->reset_horizontal_counter(); break; // TODO: audio will now be out of synchronisation — fix - break; case 0x04: - case 0x05: { - int entry = decodedAddress - 0x04; - player_and_missile_size_[entry] = *value; - missile_size_[entry] = 1 << ((*value >> 4)&3); - - uint8_t repeatMask = (*value)&7; - has_second_copy_[entry] = (repeatMask == 1) || (repeatMask == 3); - has_third_copy_[entry] = (repeatMask == 2) || (repeatMask == 3) || (repeatMask == 6); - has_fourth_copy_[entry] = (repeatMask == 4) || (repeatMask == 6); - } break; - + case 0x05: update_video(); tia_->set_player_number_and_size(decodedAddress - 0x04, *value); break; case 0x06: - case 0x07: player_colour_[decodedAddress - 0x06] = *value; break; - case 0x08: playfield_colour_ = *value; break; - case 0x09: background_colour_ = *value; break; - - case 0x0a: { - uint8_t old_playfield_control = playfield_control_; - playfield_control_ = *value; - ball_size_ = 1 << ((playfield_control_ >> 4)&3); - - // did the mirroring bit change? - if((playfield_control_^old_playfield_control)&1) { - if(playfield_control_&1) { - for(int c = 0; c < 20; c++) playfield_[c+20] = playfield_[19-c]; - } else { - memcpy(&playfield_[20], playfield_, 20); - } - } - } break; + case 0x07: update_video(); tia_->set_player_missile_colour(decodedAddress - 0x06, *value); break; + case 0x08: update_video(); tia_->set_playfield_ball_colour(*value); break; + case 0x09: update_video(); tia_->set_background_colour(*value); break; + case 0x0a: update_video(); tia_->set_playfield_control_and_ball_size(*value); break; case 0x0b: - case 0x0c: player_reflection_mask_[decodedAddress - 0x0b] = (*value)&8 ? 0 : 7; break; - + case 0x0c: update_video(); tia_->set_player_reflected(decodedAddress - 0x0b, !((*value)&8)); break; case 0x0d: - playfield_[0] = ((*value) >> 4)&1; - playfield_[1] = ((*value) >> 5)&1; - playfield_[2] = ((*value) >> 6)&1; - playfield_[3] = (*value) >> 7; - - if(playfield_control_&1) { - for(int c = 0; c < 4; c++) playfield_[39-c] = playfield_[c]; - } else { - memcpy(&playfield_[20], playfield_, 4); - } - break; case 0x0e: - playfield_[4] = (*value) >> 7; - playfield_[5] = ((*value) >> 6)&1; - playfield_[6] = ((*value) >> 5)&1; - playfield_[7] = ((*value) >> 4)&1; - playfield_[8] = ((*value) >> 3)&1; - playfield_[9] = ((*value) >> 2)&1; - playfield_[10] = ((*value) >> 1)&1; - playfield_[11] = (*value)&1; - - if(playfield_control_&1) { - for(int c = 0; c < 8; c++) playfield_[35-c] = playfield_[c+4]; - } else { - memcpy(&playfield_[24], &playfield_[4], 8); - } - break; - case 0x0f: - playfield_[19] = (*value) >> 7; - playfield_[18] = ((*value) >> 6)&1; - playfield_[17] = ((*value) >> 5)&1; - playfield_[16] = ((*value) >> 4)&1; - playfield_[15] = ((*value) >> 3)&1; - playfield_[14] = ((*value) >> 2)&1; - playfield_[13] = ((*value) >> 1)&1; - playfield_[12] = (*value)&1; - - if(playfield_control_&1) { - for(int c = 0; c < 8; c++) playfield_[27-c] = playfield_[c+12]; - } else { - memcpy(&playfield_[32], &playfield_[12], 8); - } - break; - - case 0x10: case 0x11: case 0x12: case 0x13: - case 0x14: - upcoming_events_[(upcoming_events_pointer_ + 4)%number_of_upcoming_events].updates |= Event::Action::ResetCounter; - upcoming_events_[(upcoming_events_pointer_ + 4)%number_of_upcoming_events].counter = decodedAddress - 0x10; - break; - - case 0x15: case 0x16: - update_audio(); - speaker_->set_control(decodedAddress - 0x15, *value); - break; - - case 0x17: case 0x18: - update_audio(); - speaker_->set_divider(decodedAddress - 0x17, *value); - break; - - case 0x19: case 0x1a: - update_audio(); - speaker_->set_volume(decodedAddress - 0x19, *value); - break; - - case 0x1c: - ball_graphics_enable_[1] = ball_graphics_enable_[0]; - case 0x1b: { - int index = decodedAddress - 0x1b; - player_graphics_[0][index] = *value; - player_graphics_[1][index^1] = player_graphics_[0][index^1]; - } break; + case 0x0f: update_video(); tia_->set_playfield(decodedAddress - 0x0d, *value); break; + case 0x10: + case 0x11: update_video(); tia_->set_player_position(decodedAddress - 0x10); break; + case 0x12: + case 0x13: update_video(); tia_->set_missile_position(decodedAddress - 0x13); break; + case 0x14: update_video(); tia_->set_ball_position(); break; + case 0x1b: + case 0x1c: update_video(); tia_->set_player_graphic(decodedAddress - 0x1b, *value); break; case 0x1d: - case 0x1e: - missile_graphics_enable_[decodedAddress - 0x1d] = ((*value) >> 1)&1; -// printf("e:%02x <- %c\n", decodedAddress - 0x1d, ((*value)&1) ? 'E' : '-'); - break; - case 0x1f: - ball_graphics_enable_[0] = ((*value) >> 1)&1; - break; - + case 0x1e: update_video(); tia_->set_missile_enable(decodedAddress - 0x1d, !!((*value)&2)); break; + case 0x1f: update_video(); tia_->set_ball_enable(!!((*value)&2)); break; case 0x20: - case 0x21: + case 0x21: update_video(); tia_->set_player_motion(decodedAddress - 0x20, *value); break; case 0x22: - case 0x23: - case 0x24: - object_motions_[decodedAddress - 0x20] = *value; - break; - - case 0x25: player_graphics_selector_[0] = (*value)&1; break; - case 0x26: player_graphics_selector_[1] = (*value)&1; break; - case 0x27: ball_graphics_selector_ = (*value)&1; break; - + case 0x23: update_video(); tia_->set_missile_motion(decodedAddress - 0x22, *value); break; + case 0x24: update_video(); tia_->set_ball_motion(*value); break; + case 0x25: + case 0x26: tia_->set_player_delay(decodedAddress - 0x25, !!((*value)&1)); break; + case 0x27: tia_->set_ball_delay(!!((*value)&1)); break; case 0x28: - case 0x29: - { - // TODO: this should properly mean setting a flag and propagating later, I think? - int index = decodedAddress - 0x28; - if(!(*value&0x02) && missile_graphics_reset_[index]) - { - object_counter_[object_counter_pointer_][index + 2].count = object_counter_[object_counter_pointer_][index].count; + case 0x29: update_video(); tia_->set_missile_position_to_player(decodedAddress - 0x28); break; + case 0x2a: update_video(); tia_->move(); break; + case 0x2b: update_video(); tia_->clear_motion(); break; + case 0x2c: update_video(); tia_->clear_collision_flags(); break; - uint8_t repeatMask = player_and_missile_size_[index] & 7; - int extra_offset; - switch(repeatMask) - { - default: extra_offset = 3; break; - case 5: extra_offset = 6; break; - case 7: extra_offset = 10; break; - } - - object_counter_[object_counter_pointer_][index + 2].count = (object_counter_[object_counter_pointer_][index + 2].count + extra_offset)%160; - } - missile_graphics_reset_[index] = !!((*value) & 0x02); -// printf("r:%02x <- %c\n", decodedAddress - 0x28, ((*value)&2) ? 'R' : '-'); - } - break; - - case 0x2a: { - // justification for +5: "we need to wait at least 71 [clocks] before the HMOVE operation is complete"; - // which will take 16*4 + 2 = 66 cycles from the first compare, implying the first compare must be - // in five cycles from now -// int start_pause = ((horizontal_timer_ + 3)&3) + 4; - upcoming_events_[(upcoming_events_pointer_ + 5)%number_of_upcoming_events].updates |= Event::Action::HMoveSetup; - } break; - case 0x2b: - object_motions_[0] = - object_motions_[1] = - object_motions_[2] = - object_motions_[3] = - object_motions_[4] = 0; - break; - case 0x2c: - collisions_[0] = collisions_[1] = collisions_[2] = - collisions_[3] = collisions_[4] = collisions_[5] = 0x3f; - collisions_[6] = 0x7f; - collisions_[7] = 0x3f; - break; + case 0x15: + case 0x16: update_audio(); speaker_->set_control(decodedAddress - 0x15, *value); break; + case 0x17: + case 0x18: update_audio(); speaker_->set_divider(decodedAddress - 0x17, *value); break; + case 0x19: + case 0x1a: update_audio(); speaker_->set_volume(decodedAddress - 0x19, *value); break; } } } @@ -764,7 +264,7 @@ void Machine::configure_as_target(const StaticAnalyser::Target &target) rom_pages_[3] = &rom_[3072 & romMask]; } -#pragma mark - Audio +#pragma mark - Audio and Video void Machine::update_audio() { @@ -774,9 +274,15 @@ void Machine::update_audio() cycles_since_speaker_update_ %= 114; } +void Machine::update_video() +{ + tia_->run_for_cycles((int)cycles_since_video_update_); + cycles_since_video_update_ = 0; +} + void Machine::synchronise() { update_audio(); + update_video(); speaker_->flush(); } - diff --git a/Machines/Atari2600/Atari2600.hpp b/Machines/Atari2600/Atari2600.hpp index b0320a267..d78bae584 100644 --- a/Machines/Atari2600/Atari2600.hpp +++ b/Machines/Atari2600/Atari2600.hpp @@ -15,6 +15,7 @@ #include "../CRTMachine.hpp" #include "PIA.hpp" #include "Speaker.hpp" +#include "TIA.hpp" #include "../ConfigurationTarget.hpp" #include "Atari2600Inputs.h" @@ -46,129 +47,31 @@ class Machine: // to satisfy CRTMachine::Machine virtual void setup_output(float aspect_ratio); virtual void close_output(); - virtual std::shared_ptr get_crt() { return crt_; } + virtual std::shared_ptr get_crt() { return tia_->get_crt(); } virtual std::shared_ptr get_speaker() { return speaker_; } virtual void run_for_cycles(int number_of_cycles) { CPU6502::Processor::run_for_cycles(number_of_cycles); } - // TODO: different rate for PAL private: uint8_t *rom_, *rom_pages_[4]; size_t rom_size_; - // the RIOT + // the RIOT and TIA PIA mos6532_; - - // playfield registers - uint8_t playfield_control_; - uint8_t playfield_colour_; - uint8_t background_colour_; - uint8_t playfield_[41]; - - // ... and derivatives - int ball_size_, missile_size_[2]; - - // delayed clock events - enum OutputState { - Sync, - Blank, - ColourBurst, - Pixel - }; - - struct Event { - enum Action { - Playfield = 1 << 0, - ResetCounter = 1 << 1, - - HMoveSetup = 1 << 2, - HMoveCompare = 1 << 3, - HMoveDecrement = 1 << 4, - }; - int updates; - - OutputState state; - uint8_t playfield_pixel; - int counter; - - Event() : updates(0), playfield_pixel(0) {} - } upcoming_events_[number_of_upcoming_events]; - unsigned int upcoming_events_pointer_; - - // object counters - struct ObjectCounter { - int count; // the counter value, multiplied by four, counting phase - int pixel; // for non-sprite objects, a count of cycles since the last counter reset; for sprite objects a count of pixels so far elapsed - int broad_pixel; // for sprite objects, a count of cycles since the last counter reset; otherwise unused - - ObjectCounter() : count(0), pixel(0), broad_pixel(0) {} - } object_counter_[number_of_recorded_counters][5]; - unsigned int object_counter_pointer_; - - // the latched playfield output - uint8_t playfield_output_, next_playfield_output_; - - // player registers - uint8_t player_colour_[2]; - uint8_t player_reflection_mask_[2]; - uint8_t player_graphics_[2][2]; - uint8_t player_graphics_selector_[2]; - - // object flags - bool has_second_copy_[2]; - bool has_third_copy_[2]; - bool has_fourth_copy_[2]; - uint8_t object_motions_[5]; // the value stored to this counter's motion register - - // player + missile registers - uint8_t player_and_missile_size_[2]; - - // missile registers - uint8_t missile_graphics_enable_[2]; - bool missile_graphics_reset_[2]; - - // ball registers - uint8_t ball_graphics_enable_[2]; - uint8_t ball_graphics_selector_; - - // graphics output - unsigned int horizontal_timer_; - bool vsync_enabled_, vblank_enabled_; - - // horizontal motion control - uint8_t hmove_counter_; - uint8_t hmove_flags_; + std::unique_ptr tia_; // joystick state uint8_t tia_input_value_[2]; - // collisions - uint8_t collisions_[8]; - - void output_pixels(unsigned int count); - uint8_t get_output_pixel(); - void update_timers(int mask); - // outputs - std::shared_ptr crt_; std::shared_ptr speaker_; - // current mode - bool is_pal_region_; - // speaker backlog accumlation counter unsigned int cycles_since_speaker_update_; void update_audio(); - // latched output state - unsigned int last_output_state_duration_; - OutputState state_by_extend_time_[2][57]; - OutputState *state_by_time_; - OutputState last_output_state_; - uint8_t *output_buffer_; - - // lookup table for collision reporting - uint8_t reported_collisions_[64][8]; - void setup_reported_collisions(); + // video backlog accumulation counter + unsigned int cycles_since_video_update_; + void update_video(); }; } diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index bac602803..f600468a7 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -7,3 +7,52 @@ // #include "TIA.hpp" + +using namespace Atari2600; + +TIA::TIA() +{ +/* + crt_.reset(new Outputs::CRT::CRT(228, 1, 263, Outputs::CRT::ColourSpace::YIQ, 228, 1, false, 1)); + crt_->set_output_device(Outputs::CRT::Television); + + // this is the NTSC phase offset function; see below for PAL + crt_->set_composite_sampling_function( + "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" + "{" + "uint c = texture(texID, coordinate).r;" + "uint y = c & 14u;" + "uint iPhase = (c >> 4);" + + "float phaseOffset = 6.283185308 * float(iPhase) / 13.0 + 5.074880441076923;" + "return mix(float(y) / 14.0, step(1, iPhase) * cos(phase + phaseOffset), amplitude);" + "}"); + speaker_->set_input_rate((float)(get_clock_rate() / 38.0));*/ +} + +/*void Machine::switch_region() +{ + // the PAL function + crt_->set_composite_sampling_function( + "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" + "{" + "uint c = texture(texID, coordinate).r;" + "uint y = c & 14u;" + "uint iPhase = (c >> 4);" + + "uint direction = iPhase & 1u;" + "float phaseOffset = float(7u - direction) + (float(direction) - 0.5) * 2.0 * float(iPhase >> 1);" + "phaseOffset *= 6.283185308 / 12.0;" + "return mix(float(y) / 14.0, step(4, (iPhase + 2u) & 15u) * cos(phase + phaseOffset), amplitude);" + "}"); + + crt_->set_new_timing(228, 312, Outputs::CRT::ColourSpace::YUV, 228, 1, true); + + is_pal_region_ = true; + speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); + set_clock_rate(PAL_clock_rate); +}*/ + + // justification for +5: "we need to wait at least 71 [clocks] before the HMOVE operation is complete"; + // which will take 16*4 + 2 = 66 cycles from the first compare, implying the first compare must be + // in five cycles from now diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 24d3376ad..334c87875 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -10,17 +10,22 @@ #define TIA_hpp #include +#include "../CRTMachine.hpp" namespace Atari2600 { class TIA { public: + TIA(); + ~TIA(); + void run_for_cycles(int number_of_cycles); void set_vsync(bool vsync); void set_vblank(bool vblank); - void reset_horizontal_counter(); - int get_cycles_until_horizontal_blank(); + void reset_horizontal_counter(); // Reset is delayed by four cycles. + + int get_cycles_until_horizontal_blank(unsigned int from_offset); void set_background_colour(uint8_t colour); @@ -51,6 +56,11 @@ class TIA { uint8_t get_collision_flags(int offset); void clear_collision_flags(); + + virtual std::shared_ptr get_crt() { return crt_; } + + private: + std::shared_ptr crt_; }; } From fba6baaa9c148c65f761171d739f2f9ae6354b41 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 28 Jan 2017 21:56:01 -0500 Subject: [PATCH 03/91] Stubbed and disabled to get back to building. --- Machines/Atari2600/TIA.cpp | 119 ++++++++++++++++++ .../Machine/Wrappers/CSAtari2600.mm | 2 +- 2 files changed, 120 insertions(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index f600468a7..db626bf88 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -30,6 +30,10 @@ TIA::TIA() speaker_->set_input_rate((float)(get_clock_rate() / 38.0));*/ } +TIA::~TIA() +{ +} + /*void Machine::switch_region() { // the PAL function @@ -56,3 +60,118 @@ TIA::TIA() // justification for +5: "we need to wait at least 71 [clocks] before the HMOVE operation is complete"; // which will take 16*4 + 2 = 66 cycles from the first compare, implying the first compare must be // in five cycles from now + + +void TIA::run_for_cycles(int number_of_cycles) +{ +} + +void TIA::set_vsync(bool vsync) +{ +} + +void TIA::set_vblank(bool vblank) +{ +} + +void TIA::reset_horizontal_counter() +{ +} + +int TIA::get_cycles_until_horizontal_blank(unsigned int from_offset) +{ + return 10; +} + +void TIA::set_background_colour(uint8_t colour) +{ +} + +void TIA::set_playfield(uint16_t offset, uint8_t value) +{ +} + +void TIA::set_playfield_control_and_ball_size(uint8_t value) +{ +} + +void TIA::set_playfield_ball_colour(uint8_t colour) +{ +} + +void TIA::set_player_number_and_size(int player, uint8_t value) +{ +} + +void TIA::set_player_graphic(int player, uint8_t value) +{ +} + +void TIA::set_player_reflected(int player, bool reflected) +{ +} + +void TIA::set_player_delay(int player, bool delay) +{ +} + +void TIA::set_player_position(int player) +{ +} + +void TIA::set_player_motion(int player, uint8_t motion) +{ +} + +void TIA::set_player_missile_colour(int player, uint8_t colour) +{ +} + +void TIA::set_missile_enable(int missile, bool enabled) +{ +} + +void TIA::set_missile_position(int missile) +{ +} + +void TIA::set_missile_position_to_player(int missile) +{ +} + +void TIA::set_missile_motion(int missile, uint8_t motion) +{ +} + +void TIA::set_ball_enable(bool enabled) +{ +} + +void TIA::set_ball_delay(bool delay) +{ +} + +void TIA::set_ball_position() +{ +} + +void TIA::set_ball_motion(uint8_t motion) +{ +} + +void TIA::move() +{ +} + +void TIA::clear_motion() +{ +} + +uint8_t TIA::get_collision_flags(int offset) +{ + return 0xff; +} + +void TIA::clear_collision_flags() +{ +} diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm index bf268669b..099611af8 100644 --- a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm @@ -42,7 +42,7 @@ struct CRTDelegate: public Outputs::CRT::Delegate { if(numberOfUnexpectedSyncs >= numberOfFrames >> 1) { [self.view performWithGLContext:^{ - _atari2600.switch_region(); +// _atari2600.switch_region(); }]; } } From 9c3597c7e3cf22dba8d91fe52f7eac1bc3f8ff8b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Jan 2017 13:47:36 -0500 Subject: [PATCH 04/91] Attempted to reintroduce enough logic to handle [most of] line timing, such that WSYNC works. Initial objective is to get back to having a working background. --- Machines/Atari2600/TIA.cpp | 40 ++++++++++++++++++++++++++++++++++++-- Machines/Atari2600/TIA.hpp | 4 ++++ 2 files changed, 42 insertions(+), 2 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index db626bf88..ad0db590c 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -9,8 +9,14 @@ #include "TIA.hpp" using namespace Atari2600; +namespace { + const int cycles_per_line = 228; + const double NTSC_clock_rate = 1194720; + const double PAL_clock_rate = 1182298; +} -TIA::TIA() +TIA::TIA() : + horizontal_counter_(0) { /* crt_.reset(new Outputs::CRT::CRT(228, 1, 263, Outputs::CRT::ColourSpace::YIQ, 228, 1, false, 1)); @@ -64,6 +70,28 @@ TIA::~TIA() void TIA::run_for_cycles(int number_of_cycles) { + // if part way through a line, definitely perform a partial, at most up to the end of the line + if(horizontal_counter_) + { + int cycles = std::min(number_of_cycles, cycles_per_line - horizontal_counter_); + output_for_cycles(cycles); + horizontal_counter_ = (horizontal_counter_ + cycles) % cycles_per_line; + number_of_cycles -= cycles; + } + + // output full lines for as long as possible + while(number_of_cycles > cycles_per_line) + { + output_line(); + number_of_cycles -= cycles_per_line; + } + + // partly start a new line if necessary + if(number_of_cycles) + { + output_for_cycles(number_of_cycles); + horizontal_counter_ = (horizontal_counter_ + number_of_cycles) % cycles_per_line; + } } void TIA::set_vsync(bool vsync) @@ -80,7 +108,7 @@ void TIA::reset_horizontal_counter() int TIA::get_cycles_until_horizontal_blank(unsigned int from_offset) { - return 10; + return cycles_per_line - (horizontal_counter_ + (int)from_offset) % cycles_per_line; } void TIA::set_background_colour(uint8_t colour) @@ -175,3 +203,11 @@ uint8_t TIA::get_collision_flags(int offset) void TIA::clear_collision_flags() { } + +void TIA::output_for_cycles(int number_of_cycles) +{ +} + +void TIA::output_line() +{ +} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 334c87875..153d93a39 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -61,6 +61,10 @@ class TIA { private: std::shared_ptr crt_; + + int horizontal_counter_; + void output_for_cycles(int number_of_cycles); + void output_line(); }; } From 2432a3b4d746e89e3a4c15b7b9f19fc50fc9ac03 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Jan 2017 14:00:01 -0500 Subject: [PATCH 05/91] =?UTF-8?q?Fixed=20condition=20=E2=80=94=20>=3D=20is?= =?UTF-8?q?=20smarter.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Machines/Atari2600/TIA.cpp | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index ad0db590c..9149fc867 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -18,7 +18,6 @@ namespace { TIA::TIA() : horizontal_counter_(0) { -/* crt_.reset(new Outputs::CRT::CRT(228, 1, 263, Outputs::CRT::ColourSpace::YIQ, 228, 1, false, 1)); crt_->set_output_device(Outputs::CRT::Television); @@ -33,7 +32,7 @@ TIA::TIA() : "float phaseOffset = 6.283185308 * float(iPhase) / 13.0 + 5.074880441076923;" "return mix(float(y) / 14.0, step(1, iPhase) * cos(phase + phaseOffset), amplitude);" "}"); - speaker_->set_input_rate((float)(get_clock_rate() / 38.0));*/ +/* speaker_->set_input_rate((float)(get_clock_rate() / 38.0));*/ } TIA::~TIA() @@ -80,7 +79,7 @@ void TIA::run_for_cycles(int number_of_cycles) } // output full lines for as long as possible - while(number_of_cycles > cycles_per_line) + while(number_of_cycles >= cycles_per_line) { output_line(); number_of_cycles -= cycles_per_line; From 2390358c245032c6777a1e8eb20c91a648488eda Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Jan 2017 14:19:26 -0500 Subject: [PATCH 06/91] Prevented unbounded CPU usage, albeit without yet deciding who has authority for the clock rate. --- Machines/Atari2600/Atari2600.cpp | 3 ++- Machines/Atari2600/TIA.cpp | 8 +++++--- Machines/Atari2600/TIA.hpp | 6 ++++++ 3 files changed, 13 insertions(+), 4 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index 75db7269f..cd67696a0 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -12,7 +12,6 @@ using namespace Atari2600; namespace { - static const unsigned int horizontalTimerPeriod = 228; static const double NTSC_clock_rate = 1194720; static const double PAL_clock_rate = 1182298; } @@ -31,6 +30,7 @@ void Machine::setup_output(float aspect_ratio) { tia_.reset(new TIA); speaker_.reset(new Speaker); + speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); } void Machine::close_output() @@ -56,6 +56,7 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin // skips to the end of the line. if(operation == CPU6502::BusOperation::Ready) { cycles_run_for = (unsigned int)tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_); + set_ready_line(false); } cycles_since_speaker_update_ += cycles_run_for; diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 9149fc867..df443a566 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -11,16 +11,18 @@ using namespace Atari2600; namespace { const int cycles_per_line = 228; - const double NTSC_clock_rate = 1194720; - const double PAL_clock_rate = 1182298; } TIA::TIA() : horizontal_counter_(0) { - crt_.reset(new Outputs::CRT::CRT(228, 1, 263, Outputs::CRT::ColourSpace::YIQ, 228, 1, false, 1)); + crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); crt_->set_output_device(Outputs::CRT::Television); + set_output_mode(OutputMode::NTSC); +} +void TIA::set_output_mode(Atari2600::TIA::OutputMode output_mode) +{ // this is the NTSC phase offset function; see below for PAL crt_->set_composite_sampling_function( "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 153d93a39..1f393eceb 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -19,7 +19,12 @@ class TIA { TIA(); ~TIA(); + enum class OutputMode { + NTSC, PAL + }; + void run_for_cycles(int number_of_cycles); + void set_output_mode(OutputMode output_mode); void set_vsync(bool vsync); void set_vblank(bool vblank); @@ -63,6 +68,7 @@ class TIA { std::shared_ptr crt_; int horizontal_counter_; + void output_for_cycles(int number_of_cycles); void output_line(); }; From d51f185dc77b2d73b8f8b377fa4fd42d7ed77474 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Jan 2017 15:43:57 -0500 Subject: [PATCH 07/91] Made an attempt to reintroduce the basic horizontal loop. --- Machines/Atari2600/TIA.cpp | 73 ++++++++++++++++++- Machines/Atari2600/TIA.hpp | 2 + .../xcschemes/Clock Signal.xcscheme | 1 + 3 files changed, 73 insertions(+), 3 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index df443a566..0a515bea1 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -14,7 +14,9 @@ namespace { } TIA::TIA() : - horizontal_counter_(0) + horizontal_counter_(0), + output_cursor_(0), + pixel_target_(nullptr) { crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); crt_->set_output_device(Outputs::CRT::Television); @@ -76,7 +78,6 @@ void TIA::run_for_cycles(int number_of_cycles) { int cycles = std::min(number_of_cycles, cycles_per_line - horizontal_counter_); output_for_cycles(cycles); - horizontal_counter_ = (horizontal_counter_ + cycles) % cycles_per_line; number_of_cycles -= cycles; } @@ -91,7 +92,6 @@ void TIA::run_for_cycles(int number_of_cycles) if(number_of_cycles) { output_for_cycles(number_of_cycles); - horizontal_counter_ = (horizontal_counter_ + number_of_cycles) % cycles_per_line; } } @@ -205,10 +205,77 @@ void TIA::clear_collision_flags() { } +// case 0: case 1: case 2: case 3: state = OutputState::Blank; break; +// case 4: case 5: case 6: case 7: state = OutputState::Sync; break; +// case 8: case 9: case 10: case 11: state = OutputState::ColourBurst; break; +// case 12: case 13: case 14: +// case 15: case 16: state = OutputState::Blank; break; +// +// case 17: case 18: state = vbextend ? OutputState::Blank : OutputState::Pixel; break; +// default: state = OutputState::Pixel; break; + void TIA::output_for_cycles(int number_of_cycles) { + /* + Line timing is oriented around 0 being the start of the right-hand side vertical blank; + a wsync synchronises the CPU to horizontal_counter_ = 0. All timing below is in terms of the + NTSC colour clock. + + Therefore, each line is composed of: + + 16 cycles: blank ; -> 16 + 16 cycles: sync ; -> 32 + 16 cycles: colour burst ; -> 48 + 20 cycles: blank ; -> 68 + 8 cycles: blank or pixels, depending on whether the blank extend bit is set + 152 cycles: pixels + */ + horizontal_counter_ += number_of_cycles; + if(!output_cursor_ && horizontal_counter_ >= 16) + { + crt_->output_blank(32); + output_cursor_ = 16; + } + if(output_cursor_ == 16 && horizontal_counter_ >= 32) + { + crt_->output_sync(32); + output_cursor_ = 32; + } + if(output_cursor_ == 32 && horizontal_counter_ >= 48) + { + crt_->output_default_colour_burst(32); + output_cursor_ = 48; + } + if(output_cursor_ == 48 && horizontal_counter_ >= 68) + { + crt_->output_default_colour_burst(40); + output_cursor_ = 68; + } + if(horizontal_counter_ > 68) + { + if(output_cursor_ == 68) + { + pixel_target_ = crt_->allocate_write_area(160); + } + if(pixel_target_) + { + while(output_cursor_ < horizontal_counter_) + { + pixel_target_[output_cursor_ - 68] = (output_cursor_&1) ? 0xff : 0x00; + output_cursor_++; + } + } else output_cursor_ = horizontal_counter_; + if(horizontal_counter_ == cycles_per_line) + { + crt_->output_data(320, 2); + pixel_target_ = nullptr; + } + } + horizontal_counter_ %= cycles_per_line; } void TIA::output_line() { + // TODO: optimise special case + output_for_cycles(cycles_per_line); } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 1f393eceb..b79fb3274 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -68,6 +68,8 @@ class TIA { std::shared_ptr crt_; int horizontal_counter_; + int output_cursor_; + uint8_t *pixel_target_; void output_for_cycles(int number_of_cycles); void output_line(); 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 6cceb313b..08fb44b13 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -69,6 +69,7 @@ useCustomWorkingDirectory = "NO" ignoresPersistentStateOnLaunch = "NO" debugDocumentVersioning = "YES" + enableAddressSanitizer = "YES" debugServiceExtension = "internal" allowLocationSimulation = "NO"> Date: Sun, 29 Jan 2017 16:11:29 -0500 Subject: [PATCH 08/91] Added an extra flag to avoid potential race condition on is_full_, being reset from the background despite a write area not having been allocated. --- Outputs/CRT/Internals/TextureBuilder.cpp | 5 ++++- Outputs/CRT/Internals/TextureBuilder.hpp | 1 + 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/Outputs/CRT/Internals/TextureBuilder.cpp b/Outputs/CRT/Internals/TextureBuilder.cpp index 29c0f87b1..89efc616e 100644 --- a/Outputs/CRT/Internals/TextureBuilder.cpp +++ b/Outputs/CRT/Internals/TextureBuilder.cpp @@ -43,6 +43,7 @@ TextureBuilder::TextureBuilder(size_t bytes_per_pixel, GLenum texture_unit) : write_areas_start_y_(0), is_full_(false), did_submit_(false), + has_write_area_(false), number_of_write_areas_(0) { image_.resize(bytes_per_pixel * InputBufferBuilderWidth * InputBufferBuilderHeight); @@ -105,6 +106,7 @@ uint8_t *TextureBuilder::allocate_write_area(size_t required_length) else write_areas_.push_back(next_write_area); number_of_write_areas_++; + has_write_area_ = true; return pointer_to_location(next_write_area.x, next_write_area.y); } @@ -116,8 +118,9 @@ bool TextureBuilder::is_full() void TextureBuilder::reduce_previous_allocation_to(size_t actual_length) { - if(is_full_) return; + if(is_full_ || !has_write_area_) return; + has_write_area_ = false; WriteArea &write_area = write_areas_[number_of_write_areas_-1]; write_area.length = (uint16_t)actual_length; diff --git a/Outputs/CRT/Internals/TextureBuilder.hpp b/Outputs/CRT/Internals/TextureBuilder.hpp index d200a9b97..93777ac92 100644 --- a/Outputs/CRT/Internals/TextureBuilder.hpp +++ b/Outputs/CRT/Internals/TextureBuilder.hpp @@ -69,6 +69,7 @@ class TextureBuilder { size_t number_of_write_areas_; bool is_full_; bool did_submit_; + bool has_write_area_; inline uint8_t *pointer_to_location(uint16_t x, uint16_t y); // Usually: the start position for the current batch of write areas. From ebde9553566a222b4f8be6e47cbc4e723c172f92 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Jan 2017 16:12:48 -0500 Subject: [PATCH 09/91] This needs to be a `memmove` as the areas may overlap. --- Outputs/CRT/Internals/ArrayBuilder.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Outputs/CRT/Internals/ArrayBuilder.cpp b/Outputs/CRT/Internals/ArrayBuilder.cpp index 0793dc372..afda1c9b6 100644 --- a/Outputs/CRT/Internals/ArrayBuilder.cpp +++ b/Outputs/CRT/Internals/ArrayBuilder.cpp @@ -130,7 +130,7 @@ void ArrayBuilder::Buffer::flush() { if(submitted_data) { - memcpy(data.data(), &data[submitted_data], allocated_data - submitted_data); + memmove(data.data(), &data[submitted_data], allocated_data - submitted_data); allocated_data -= submitted_data; flushed_data -= submitted_data; submitted_data = 0; From 92754ace7a03f9df256415525ab403bc55f03e2e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 29 Jan 2017 22:16:23 -0500 Subject: [PATCH 10/91] Some mild fixes get me up to having a rolling screen of vertical lines. Which is what I was hoping for right now! --- Machines/Atari2600/Atari2600.cpp | 4 +- Machines/Atari2600/TIA.cpp | 43 ++++++++++++++++--- Machines/Atari2600/TIA.hpp | 8 +++- .../xcschemes/Clock Signal.xcscheme | 1 - 4 files changed, 45 insertions(+), 11 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index cd67696a0..7f7f70ea6 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -130,8 +130,8 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin } else { const uint16_t decodedAddress = address & 0x3f; switch(decodedAddress) { - case 0x00: update_video(); tia_->set_vsync(!!(*value & 0x02)); break; - case 0x01: update_video(); tia_->set_vblank(!!(*value & 0x02)); break; + case 0x00: update_video(); tia_->set_sync(!!(*value & 0x02)); break; + case 0x01: update_video(); tia_->set_blank(!!(*value & 0x02)); break; case 0x02: set_ready_line(true); diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 0a515bea1..2dbdd4c03 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -11,12 +11,17 @@ using namespace Atari2600; namespace { const int cycles_per_line = 228; + + const int sync_flag = 0x1; + const int blank_flag = 0x2; } TIA::TIA() : horizontal_counter_(0), output_cursor_(0), - pixel_target_(nullptr) + pixel_target_(nullptr), + requested_output_mode_(0), + output_mode_(0) { crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); crt_->set_output_device(Outputs::CRT::Television); @@ -95,12 +100,14 @@ void TIA::run_for_cycles(int number_of_cycles) } } -void TIA::set_vsync(bool vsync) +void TIA::set_sync(bool sync) { + requested_output_mode_ = (requested_output_mode_ & ~sync_flag) | (sync ? sync_flag : 0); } -void TIA::set_vblank(bool vblank) +void TIA::set_blank(bool blank) { + requested_output_mode_ = (requested_output_mode_ & ~blank_flag) | (blank ? blank_flag : 0); } void TIA::reset_horizontal_counter() @@ -230,7 +237,12 @@ void TIA::output_for_cycles(int number_of_cycles) 8 cycles: blank or pixels, depending on whether the blank extend bit is set 152 cycles: pixels */ +// if(output_mode_ != requested_output_mode_) +// { +// // flush the old output +// } horizontal_counter_ += number_of_cycles; + if(!output_cursor_ && horizontal_counter_ >= 16) { crt_->output_blank(32); @@ -248,7 +260,7 @@ void TIA::output_for_cycles(int number_of_cycles) } if(output_cursor_ == 48 && horizontal_counter_ >= 68) { - crt_->output_default_colour_burst(40); + crt_->output_blank(40); output_cursor_ = 68; } if(horizontal_counter_ > 68) @@ -272,10 +284,29 @@ void TIA::output_for_cycles(int number_of_cycles) } } horizontal_counter_ %= cycles_per_line; + output_cursor_ %= cycles_per_line; } void TIA::output_line() { - // TODO: optimise special case - output_for_cycles(cycles_per_line); + output_mode_ = requested_output_mode_; + switch(output_mode_) + { + default: + // TODO: optimise special case + output_for_cycles(cycles_per_line); + break; + case sync_flag: + case sync_flag | blank_flag: + crt_->output_sync(32); + crt_->output_blank(32); + crt_->output_sync(392); + break; + case blank_flag: + crt_->output_blank(32); + crt_->output_sync(32); + crt_->output_default_colour_burst(32); + crt_->output_blank(360); + break; + } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index b79fb3274..e5ec60a17 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -26,8 +26,8 @@ class TIA { void run_for_cycles(int number_of_cycles); void set_output_mode(OutputMode output_mode); - void set_vsync(bool vsync); - void set_vblank(bool vblank); + void set_sync(bool sync); + void set_blank(bool blank); void reset_horizontal_counter(); // Reset is delayed by four cycles. int get_cycles_until_horizontal_blank(unsigned int from_offset); @@ -68,7 +68,11 @@ class TIA { std::shared_ptr crt_; int horizontal_counter_; + int output_cursor_; + int output_mode_; + int requested_output_mode_; + uint8_t *pixel_target_; void output_for_cycles(int number_of_cycles); 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 08fb44b13..6cceb313b 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -69,7 +69,6 @@ useCustomWorkingDirectory = "NO" ignoresPersistentStateOnLaunch = "NO" debugDocumentVersioning = "YES" - enableAddressSanitizer = "YES" debugServiceExtension = "internal" allowLocationSimulation = "NO"> Date: Mon, 30 Jan 2017 07:19:19 -0500 Subject: [PATCH 11/91] Made a first attempt at switching to a model that respects blank and sync. --- Machines/Atari2600/TIA.cpp | 78 ++++++++++++++++++++++++-------------- Machines/Atari2600/TIA.hpp | 2 +- 2 files changed, 50 insertions(+), 30 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 2dbdd4c03..c1d74eb5d 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -20,7 +20,6 @@ TIA::TIA() : horizontal_counter_(0), output_cursor_(0), pixel_target_(nullptr), - requested_output_mode_(0), output_mode_(0) { crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); @@ -102,12 +101,12 @@ void TIA::run_for_cycles(int number_of_cycles) void TIA::set_sync(bool sync) { - requested_output_mode_ = (requested_output_mode_ & ~sync_flag) | (sync ? sync_flag : 0); + output_mode_ = (output_mode_ & ~sync_flag) | (sync ? sync_flag : 0); } void TIA::set_blank(bool blank) { - requested_output_mode_ = (requested_output_mode_ & ~blank_flag) | (blank ? blank_flag : 0); + output_mode_ = (output_mode_ & ~blank_flag) | (blank ? blank_flag : 0); } void TIA::reset_horizontal_counter() @@ -237,37 +236,58 @@ void TIA::output_for_cycles(int number_of_cycles) 8 cycles: blank or pixels, depending on whether the blank extend bit is set 152 cycles: pixels */ -// if(output_mode_ != requested_output_mode_) -// { -// // flush the old output -// } horizontal_counter_ += number_of_cycles; - if(!output_cursor_ && horizontal_counter_ >= 16) - { - crt_->output_blank(32); - output_cursor_ = 16; +#define Period(function, target) \ + if(output_cursor_ < target) \ + { \ + if(horizontal_counter_ <= target) \ + { \ + crt_->function((unsigned int)((horizontal_counter_ - output_cursor_) * 2)); \ + output_cursor_ = horizontal_counter_; \ + return; \ + } \ + else \ + { \ + crt_->function((unsigned int)((target - output_cursor_) * 2)); \ + output_cursor_ = target; \ + } \ } - if(output_cursor_ == 16 && horizontal_counter_ >= 32) + + switch(output_mode_) { - crt_->output_sync(32); - output_cursor_ = 32; + default: + Period(output_blank, 16) + Period(output_sync, 32) + Period(output_default_colour_burst, 48) + Period(output_blank, 68) + break; + case sync_flag: + case sync_flag | blank_flag: + Period(output_sync, 16) + Period(output_blank, 32) + Period(output_default_colour_burst, 48) + Period(output_sync, 228) + break; } - if(output_cursor_ == 32 && horizontal_counter_ >= 48) + + if(output_mode_ & blank_flag) { - crt_->output_default_colour_burst(32); - output_cursor_ = 48; - } - if(output_cursor_ == 48 && horizontal_counter_ >= 68) - { - crt_->output_blank(40); - output_cursor_ = 68; - } - if(horizontal_counter_ > 68) - { - if(output_cursor_ == 68) + if(pixel_target_) { - pixel_target_ = crt_->allocate_write_area(160); + crt_->output_data((unsigned int)((horizontal_counter_ - pixel_target_origin_) * 2), 2); + pixel_target_ = nullptr; + } + int duration = std::min(228, horizontal_counter_) - output_cursor_; + crt_->output_blank((unsigned int)(duration * 2)); + output_cursor_ += duration; + } + else + { + if(!pixel_target_) + { + pixel_target_ = crt_->allocate_write_area((unsigned int)(228 - output_cursor_)); + pixel_target_origin_ = output_cursor_; } if(pixel_target_) { @@ -279,17 +299,17 @@ void TIA::output_for_cycles(int number_of_cycles) } else output_cursor_ = horizontal_counter_; if(horizontal_counter_ == cycles_per_line) { - crt_->output_data(320, 2); + crt_->output_data((unsigned int)((horizontal_counter_ - pixel_target_origin_) * 2), 2); pixel_target_ = nullptr; } } + horizontal_counter_ %= cycles_per_line; output_cursor_ %= cycles_per_line; } void TIA::output_line() { - output_mode_ = requested_output_mode_; switch(output_mode_) { default: diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index e5ec60a17..f0f547d65 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -71,9 +71,9 @@ class TIA { int output_cursor_; int output_mode_; - int requested_output_mode_; uint8_t *pixel_target_; + int pixel_target_origin_; void output_for_cycles(int number_of_cycles); void output_line(); From 2b08758b2bd16795c115c1a4170f62454df5d3c3 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 30 Jan 2017 08:08:03 -0500 Subject: [PATCH 12/91] Started capturing playfield/ball and background colours. --- Machines/Atari2600/TIA.cpp | 4 +++- Machines/Atari2600/TIA.hpp | 3 +++ 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index c1d74eb5d..b11bc9d7b 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -120,6 +120,7 @@ int TIA::get_cycles_until_horizontal_blank(unsigned int from_offset) void TIA::set_background_colour(uint8_t colour) { + background_colour_ = colour; } void TIA::set_playfield(uint16_t offset, uint8_t value) @@ -132,6 +133,7 @@ void TIA::set_playfield_control_and_ball_size(uint8_t value) void TIA::set_playfield_ball_colour(uint8_t colour) { + playfield_ball_colour_ = colour; } void TIA::set_player_number_and_size(int player, uint8_t value) @@ -293,7 +295,7 @@ void TIA::output_for_cycles(int number_of_cycles) { while(output_cursor_ < horizontal_counter_) { - pixel_target_[output_cursor_ - 68] = (output_cursor_&1) ? 0xff : 0x00; + pixel_target_[output_cursor_ - 68] = (output_cursor_&1) ? playfield_ball_colour_ : background_colour_; output_cursor_++; } } else output_cursor_ = horizontal_counter_; diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index f0f547d65..98cb6ab68 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -75,6 +75,9 @@ class TIA { uint8_t *pixel_target_; int pixel_target_origin_; + uint8_t playfield_ball_colour_; + uint8_t background_colour_; + void output_for_cycles(int number_of_cycles); void output_line(); }; From 8545707b5492c98b25789653478cdaaeab534fe5 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 30 Jan 2017 21:38:58 -0500 Subject: [PATCH 13/91] Reinstituted the playfield. Probably needs more buffering though. Time to look into delays. --- Machines/Atari2600/TIA.cpp | 33 +++++++++++++++++++++++++++++++-- Machines/Atari2600/TIA.hpp | 2 ++ 2 files changed, 33 insertions(+), 2 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index b11bc9d7b..04e7cabc6 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -14,17 +14,29 @@ namespace { const int sync_flag = 0x1; const int blank_flag = 0x2; + + uint8_t reverse_table[256]; } TIA::TIA() : horizontal_counter_(0), output_cursor_(0), pixel_target_(nullptr), - output_mode_(0) + output_mode_(0), + background_{0, 0}, + background_half_mask_(0) { crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); crt_->set_output_device(Outputs::CRT::Television); set_output_mode(OutputMode::NTSC); + + for(int c = 0; c < 256; c++) + { + reverse_table[c] = (uint8_t)( + ((c & 0x01) << 7) | ((c & 0x02) << 5) | ((c & 0x04) << 3) | ((c & 0x08) << 1) | + ((c & 0x10) >> 1) | ((c & 0x20) >> 3) | ((c & 0x40) >> 5) | ((c & 0x80) >> 7) + ); + } } void TIA::set_output_mode(Atari2600::TIA::OutputMode output_mode) @@ -125,10 +137,26 @@ void TIA::set_background_colour(uint8_t colour) void TIA::set_playfield(uint16_t offset, uint8_t value) { + switch(offset) + { + case 0: + background_[1] = (background_[1] & 0x0ffff) | ((uint32_t)reverse_table[value & 0xf0] << 16); + background_[0] = (background_[0] & 0xffff0) | (uint32_t)(value >> 4); + break; + case 1: + background_[1] = (background_[1] & 0xf00ff) | ((uint32_t)value << 8); + background_[0] = (background_[0] & 0xff00f) | ((uint32_t)reverse_table[value] << 4); + break; + case 2: + background_[1] = (background_[1] & 0xfff00) | reverse_table[value]; + background_[0] = (background_[0] & 0x00fff) | ((uint32_t)value << 12); + break; + } } void TIA::set_playfield_control_and_ball_size(uint8_t value) { + background_half_mask_ = value & 1; } void TIA::set_playfield_ball_colour(uint8_t colour) @@ -295,7 +323,8 @@ void TIA::output_for_cycles(int number_of_cycles) { while(output_cursor_ < horizontal_counter_) { - pixel_target_[output_cursor_ - 68] = (output_cursor_&1) ? playfield_ball_colour_ : background_colour_; + int offset = (output_cursor_ - 68) >> 2; + pixel_target_[output_cursor_ - pixel_target_origin_] = ((background_[(offset/20)&background_half_mask_] >> (offset%20))&1) ? playfield_ball_colour_ : background_colour_; output_cursor_++; } } else output_cursor_ = horizontal_counter_; diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 98cb6ab68..06f14571b 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -77,6 +77,8 @@ class TIA { uint8_t playfield_ball_colour_; uint8_t background_colour_; + uint32_t background_[2]; + int background_half_mask_; void output_for_cycles(int number_of_cycles); void output_line(); From abe04334c2031e9b52ea5386c932bdbdf3aa6f6e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 30 Jan 2017 22:42:27 -0500 Subject: [PATCH 14/91] Attempted to retain more player information, and removed the output cursor from class storage as I think it's acceptable as a temporary. --- Machines/Atari2600/TIA.cpp | 60 +++++++++++++++++++++++++++----------- Machines/Atari2600/TIA.hpp | 25 ++++++++++++++-- 2 files changed, 65 insertions(+), 20 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 04e7cabc6..9554ff97c 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -20,7 +20,6 @@ namespace { TIA::TIA() : horizontal_counter_(0), - output_cursor_(0), pixel_target_(nullptr), output_mode_(0), background_{0, 0}, @@ -166,30 +165,57 @@ void TIA::set_playfield_ball_colour(uint8_t colour) void TIA::set_player_number_and_size(int player, uint8_t value) { + switch(value & 7) + { + case 0: case 1: case 2: case 3: case 4: + player_[player].size = 0; + player_[player].copy_flags = value & 7; + break; + case 5: + player_[player].size = 1; + player_[player].copy_flags = 0; + break; + case 6: + player_[player].size = 0; + player_[player].copy_flags = 7; + break; + case 7: + player_[player].size = 2; + player_[player].copy_flags = 0; + break; + } + + missile_[player].size = (value >> 4)&3; } void TIA::set_player_graphic(int player, uint8_t value) { + player_[player].graphic = value; } void TIA::set_player_reflected(int player, bool reflected) { + player_[player].reverse_mask = reflected ? 7 : 0; } void TIA::set_player_delay(int player, bool delay) { + // TODO } void TIA::set_player_position(int player) { + // TODO } void TIA::set_player_motion(int player, uint8_t motion) { + player_[player].motion = motion >> 4; } void TIA::set_player_missile_colour(int player, uint8_t colour) { + player_[player].colour = colour; } void TIA::set_missile_enable(int missile, bool enabled) @@ -234,7 +260,7 @@ void TIA::clear_motion() uint8_t TIA::get_collision_flags(int offset) { - return 0xff; + return 0x00; } void TIA::clear_collision_flags() @@ -266,21 +292,22 @@ void TIA::output_for_cycles(int number_of_cycles) 8 cycles: blank or pixels, depending on whether the blank extend bit is set 152 cycles: pixels */ + int output_cursor = horizontal_counter_; horizontal_counter_ += number_of_cycles; #define Period(function, target) \ - if(output_cursor_ < target) \ + if(output_cursor < target) \ { \ if(horizontal_counter_ <= target) \ { \ - crt_->function((unsigned int)((horizontal_counter_ - output_cursor_) * 2)); \ - output_cursor_ = horizontal_counter_; \ + crt_->function((unsigned int)((horizontal_counter_ - output_cursor) * 2)); \ + output_cursor = horizontal_counter_; \ return; \ } \ else \ { \ - crt_->function((unsigned int)((target - output_cursor_) * 2)); \ - output_cursor_ = target; \ + crt_->function((unsigned int)((target - output_cursor) * 2)); \ + output_cursor = target; \ } \ } @@ -308,26 +335,26 @@ void TIA::output_for_cycles(int number_of_cycles) crt_->output_data((unsigned int)((horizontal_counter_ - pixel_target_origin_) * 2), 2); pixel_target_ = nullptr; } - int duration = std::min(228, horizontal_counter_) - output_cursor_; + int duration = std::min(228, horizontal_counter_) - output_cursor; crt_->output_blank((unsigned int)(duration * 2)); - output_cursor_ += duration; + output_cursor += duration; } else { if(!pixel_target_) { - pixel_target_ = crt_->allocate_write_area((unsigned int)(228 - output_cursor_)); - pixel_target_origin_ = output_cursor_; + pixel_target_ = crt_->allocate_write_area((unsigned int)(228 - output_cursor)); + pixel_target_origin_ = output_cursor; } if(pixel_target_) { - while(output_cursor_ < horizontal_counter_) + while(output_cursor < horizontal_counter_) { - int offset = (output_cursor_ - 68) >> 2; - pixel_target_[output_cursor_ - pixel_target_origin_] = ((background_[(offset/20)&background_half_mask_] >> (offset%20))&1) ? playfield_ball_colour_ : background_colour_; - output_cursor_++; + int offset = (output_cursor - 68) >> 2; + pixel_target_[output_cursor - pixel_target_origin_] = ((background_[(offset/20)&background_half_mask_] >> (offset%20))&1) ? playfield_ball_colour_ : background_colour_; + output_cursor++; } - } else output_cursor_ = horizontal_counter_; + } else output_cursor = horizontal_counter_; if(horizontal_counter_ == cycles_per_line) { crt_->output_data((unsigned int)((horizontal_counter_ - pixel_target_origin_) * 2), 2); @@ -336,7 +363,6 @@ void TIA::output_for_cycles(int number_of_cycles) } horizontal_counter_ %= cycles_per_line; - output_cursor_ %= cycles_per_line; } void TIA::output_line() diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 06f14571b..3883fe0bc 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -67,21 +67,40 @@ class TIA { private: std::shared_ptr crt_; + // drawing methods + void output_for_cycles(int number_of_cycles); + void output_line(); + + // the master counter; counts from 0 to 228 with all visible pixels being in the final 160 int horizontal_counter_; - int output_cursor_; + // contains flags to indicate whether sync or blank are currently active int output_mode_; + // keeps track of the target pixel buffer for this line and when it was acquired uint8_t *pixel_target_; int pixel_target_origin_; + // playfield state uint8_t playfield_ball_colour_; uint8_t background_colour_; uint32_t background_[2]; int background_half_mask_; - void output_for_cycles(int number_of_cycles); - void output_line(); + // player state + struct Player { + int size; // 0 = normal, 1 = double, 2 = quad + int copy_flags; // a bit field, corresponding to the first few values of NUSIZ + uint8_t graphic; // the player graphic + uint8_t colour; // the player colour + int reverse_mask; // 7 for a reflected player, 0 for normal + uint8_t motion; // low four bits used + } player_[2]; + + // missile state + struct Missile { + int size; // 0 = 1 pixel, 1 = 2 pixels, etc + } missile_[2]; }; } From f2437cb257587eb6a30f6efb234e923dd7d0383a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 31 Jan 2017 20:30:32 -0500 Subject: [PATCH 15/91] Added some additional documentation, started making steps towards returning sprites, fixed a counter bug that would exhibit as incorrect sync. --- Machines/Atari2600/TIA.cpp | 16 +++------------- Machines/Atari2600/TIA.hpp | 18 +++++++++++++++--- 2 files changed, 18 insertions(+), 16 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 9554ff97c..8398e8ca0 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -205,7 +205,7 @@ void TIA::set_player_delay(int player, bool delay) void TIA::set_player_position(int player) { - // TODO + player_[player].position = ((horizontal_counter_ - 68) + 6)%160; } void TIA::set_player_motion(int player, uint8_t motion) @@ -267,15 +267,6 @@ void TIA::clear_collision_flags() { } -// case 0: case 1: case 2: case 3: state = OutputState::Blank; break; -// case 4: case 5: case 6: case 7: state = OutputState::Sync; break; -// case 8: case 9: case 10: case 11: state = OutputState::ColourBurst; break; -// case 12: case 13: case 14: -// case 15: case 16: state = OutputState::Blank; break; -// -// case 17: case 18: state = vbextend ? OutputState::Blank : OutputState::Pixel; break; -// default: state = OutputState::Pixel; break; - void TIA::output_for_cycles(int number_of_cycles) { /* @@ -284,7 +275,7 @@ void TIA::output_for_cycles(int number_of_cycles) NTSC colour clock. Therefore, each line is composed of: - + 16 cycles: blank ; -> 16 16 cycles: sync ; -> 32 16 cycles: colour burst ; -> 48 @@ -301,7 +292,7 @@ void TIA::output_for_cycles(int number_of_cycles) if(horizontal_counter_ <= target) \ { \ crt_->function((unsigned int)((horizontal_counter_ - output_cursor) * 2)); \ - output_cursor = horizontal_counter_; \ + horizontal_counter_ %= cycles_per_line; \ return; \ } \ else \ @@ -337,7 +328,6 @@ void TIA::output_for_cycles(int number_of_cycles) } int duration = std::min(228, horizontal_counter_) - output_cursor; crt_->output_blank((unsigned int)(duration * 2)); - output_cursor += duration; } else { diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 3883fe0bc..6c76fc9cd 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -68,8 +68,12 @@ class TIA { std::shared_ptr crt_; // drawing methods - void output_for_cycles(int number_of_cycles); - void output_line(); + inline void output_for_cycles(int number_of_cycles); + inline void output_line(); + + inline void draw_background(uint8_t *target, int start, int length); + inline void draw_playfield(uint8_t *target, int start, int length); + inline void draw_background_and_playfield(uint8_t *target, int start, int length); // the master counter; counts from 0 to 228 with all visible pixels being in the final 160 int horizontal_counter_; @@ -84,8 +88,14 @@ class TIA { // playfield state uint8_t playfield_ball_colour_; uint8_t background_colour_; - uint32_t background_[2]; int background_half_mask_; + uint32_t background_[2]; // contains two 20-bit bitfields representing the background state; + // at index 0 is the left-hand side of the playfield with bit 0 being + // the first bit to display, bit 1 the second, etc. Index 1 contains + // a mirror image of index 0. If the playfield is being displayed in + // mirroring mode, background_[0] will be output on the left and + // background_[1] on the right; otherwise background_[0] will be + // output twice. // player state struct Player { @@ -95,6 +105,8 @@ class TIA { uint8_t colour; // the player colour int reverse_mask; // 7 for a reflected player, 0 for normal uint8_t motion; // low four bits used + uint8_t position; // in the range [0, 160) to indicate offset from the left margin, i.e. phase difference + // between the player counter and the background pixel counter. } player_[2]; // missile state From 23f3ccd77a8d7af6307a6edbd75ad6313d70ce9a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 5 Feb 2017 17:47:34 -0500 Subject: [PATCH 16/91] Made a further attempt to prevent overwrites. --- Outputs/CRT/Internals/TextureBuilder.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/Outputs/CRT/Internals/TextureBuilder.cpp b/Outputs/CRT/Internals/TextureBuilder.cpp index 89efc616e..a8188b9dc 100644 --- a/Outputs/CRT/Internals/TextureBuilder.cpp +++ b/Outputs/CRT/Internals/TextureBuilder.cpp @@ -196,5 +196,6 @@ void TextureBuilder::flush(const std::function } did_submit_ = false; + has_write_area_ = false; number_of_write_areas_ = 0; } From 6bcf95042cde0cfbd26b152cf09a473fcfa06791 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 5 Feb 2017 17:51:56 -0500 Subject: [PATCH 17/91] Started trying to be a bit more explicit about usage, and to divide up drawing responsibility. --- Machines/Atari2600/TIA.cpp | 30 +++++++++++++++++++++++------- Machines/Atari2600/TIA.hpp | 23 +++++++++++++++++++---- 2 files changed, 42 insertions(+), 11 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 8398e8ca0..6a8888cac 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -126,7 +126,7 @@ void TIA::reset_horizontal_counter() int TIA::get_cycles_until_horizontal_blank(unsigned int from_offset) { - return cycles_per_line - (horizontal_counter_ + (int)from_offset) % cycles_per_line; + return 4 + (cycles_per_line - (horizontal_counter_ + (int)from_offset) % cycles_per_line); } void TIA::set_background_colour(uint8_t colour) @@ -156,6 +156,8 @@ void TIA::set_playfield(uint16_t offset, uint8_t value) void TIA::set_playfield_control_and_ball_size(uint8_t value) { background_half_mask_ = value & 1; + playfield_is_above_players_ = !!(value & 4); + playfield_is_in_score_mode_ = !playfield_is_above_players_ && (value & 2); } void TIA::set_playfield_ball_colour(uint8_t colour) @@ -338,12 +340,8 @@ void TIA::output_for_cycles(int number_of_cycles) } if(pixel_target_) { - while(output_cursor < horizontal_counter_) - { - int offset = (output_cursor - 68) >> 2; - pixel_target_[output_cursor - pixel_target_origin_] = ((background_[(offset/20)&background_half_mask_] >> (offset%20))&1) ? playfield_ball_colour_ : background_colour_; - output_cursor++; - } + draw_background(pixel_target_, output_cursor, horizontal_counter_); + output_cursor = horizontal_counter_; } else output_cursor = horizontal_counter_; if(horizontal_counter_ == cycles_per_line) { @@ -377,3 +375,21 @@ void TIA::output_line() break; } } + +#pragma mark - Background and playfield + +void TIA::draw_background(uint8_t *target, int start, int length) const +{ + if(!target) return; + int position = start; + while(length--) + { + int offset = (position - 68) >> 2; + target[position - pixel_target_origin_] = ((background_[(offset/20)&background_half_mask_] >> (offset%20))&1) ? playfield_ball_colour_ : background_colour_; + position++; + } +} + +void TIA::draw_playfield(uint8_t *target, int start, int length) const +{ +} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 6c76fc9cd..b9e56379e 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -23,6 +23,10 @@ class TIA { NTSC, PAL }; + /*! + Advances the TIA by @c number_of_cycles cycles. Any queued setters take effect in the + first cycle performed. + */ void run_for_cycles(int number_of_cycles); void set_output_mode(OutputMode output_mode); @@ -71,9 +75,8 @@ class TIA { inline void output_for_cycles(int number_of_cycles); inline void output_line(); - inline void draw_background(uint8_t *target, int start, int length); - inline void draw_playfield(uint8_t *target, int start, int length); - inline void draw_background_and_playfield(uint8_t *target, int start, int length); + inline void draw_background(uint8_t *target, int start, int length) const; + inline void draw_playfield(uint8_t *target, int start, int length) const; // the master counter; counts from 0 to 228 with all visible pixels being in the final 160 int horizontal_counter_; @@ -81,14 +84,25 @@ class TIA { // contains flags to indicate whether sync or blank are currently active int output_mode_; - // keeps track of the target pixel buffer for this line and when it was acquired + // keeps track of the target pixel buffer for this line and when it was acquired, and a corresponding collision buffer uint8_t *pixel_target_; int pixel_target_origin_; + uint8_t collision_buffer_[160]; + enum class CollisionType : uint8_t { + Playfield, + Sprite1, + Sprite2, + Missile1, + Missile2, + Ball + }; // playfield state uint8_t playfield_ball_colour_; uint8_t background_colour_; int background_half_mask_; + bool playfield_is_in_score_mode_; + bool playfield_is_above_players_; uint32_t background_[2]; // contains two 20-bit bitfields representing the background state; // at index 0 is the left-hand side of the playfield with bit 0 being // the first bit to display, bit 1 the second, etc. Index 1 contains @@ -96,6 +110,7 @@ class TIA { // mirroring mode, background_[0] will be output on the left and // background_[1] on the right; otherwise background_[0] will be // output twice. + int latched_playfield_value_; // player state struct Player { From fcf8cafb5d917f363eceae66581b69618f39121d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 6 Feb 2017 18:27:44 -0500 Subject: [PATCH 18/91] Sought to ensure that communicating a colour burst in multiple parts doesn't ruin the phase. --- Outputs/CRT/CRT.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Outputs/CRT/CRT.cpp b/Outputs/CRT/CRT.cpp index f1d0c3f88..550a9d624 100644 --- a/Outputs/CRT/CRT.cpp +++ b/Outputs/CRT/CRT.cpp @@ -222,6 +222,7 @@ void CRT::advance_cycles(unsigned int number_of_cycles, bool hsync_requested, bo (*(uint16_t *)&input_buffer[position + SourceVertexOffsetOfOutputStart + 2]) = output_y; } }); + colour_burst_amplitude_ = 0; } is_writing_composite_run_ ^= true; } @@ -284,7 +285,7 @@ void CRT::output_scan(const Scan *const scan) // simplified colour burst logic: if it's within the back porch we'll take it if(scan->type == Scan::Type::ColourBurst) { - if(horizontal_flywheel_->get_current_time() < (horizontal_flywheel_->get_standard_period() * 12) >> 6) + if(!colour_burst_amplitude_ && horizontal_flywheel_->get_current_time() < (horizontal_flywheel_->get_standard_period() * 12) >> 6) { unsigned int position_phase = (horizontal_flywheel_->get_current_time() * colour_cycle_numerator_ * 256) / phase_denominator_; colour_burst_phase_ = (position_phase + scan->phase) & 255; From 66bcdd36f3e3f52ee3210e723aa61c14a76cb82f Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 6 Feb 2017 18:29:00 -0500 Subject: [PATCH 19/91] Made an attempt to introduce an intermediate buffer that ends up with a bit mask of all graphical components present on it, and to use that to infer collision flags and colours, based on playfield priority and colour palette. Immediately yielding: a blank screen. Good work! --- Machines/Atari2600/TIA.cpp | 171 ++++++++++++++++++++++++++++++------- Machines/Atari2600/TIA.hpp | 50 +++++++---- 2 files changed, 174 insertions(+), 47 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 6a8888cac..7912efae8 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -11,6 +11,7 @@ using namespace Atari2600; namespace { const int cycles_per_line = 228; + const int first_pixel_cycle = 68; const int sync_flag = 0x1; const int blank_flag = 0x2; @@ -20,7 +21,7 @@ namespace { TIA::TIA() : horizontal_counter_(0), - pixel_target_(nullptr), + pixels_start_location_(0), output_mode_(0), background_{0, 0}, background_half_mask_(0) @@ -36,6 +37,65 @@ TIA::TIA() : ((c & 0x10) >> 1) | ((c & 0x20) >> 3) | ((c & 0x40) >> 5) | ((c & 0x80) >> 7) ); } + + for(int c = 0; c < 64; c++) + { + collision_flags_by_buffer_vaules_[c] = 0; // TODO + } + + for(int c = 0; c < 64; c++) + { + bool has_playfield = c & (int)(CollisionType::Playfield); + bool has_ball = c & (int)(CollisionType::Ball); + bool has_player0 = c & (int)(CollisionType::Player0); + bool has_player1 = c & (int)(CollisionType::Player1); + bool has_missile0 = c & (int)(CollisionType::Missile0); + bool has_missile1 = c & (int)(CollisionType::Missile1); + + colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::Background; + + if(has_playfield || has_ball) + { + colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = (uint8_t)ColourIndex::PlayfieldBall; + } + if(has_ball) + { + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = (uint8_t)ColourIndex::PlayfieldBall; + } + + if(has_player1 || has_missile1) + { + colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayerMissile1; + } + if(has_playfield) + { + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = (uint8_t)ColourIndex::PlayerMissile1; + } + + if(has_player0 || has_missile0) + { + colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = + colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayerMissile0; + } + if(has_playfield) + { + colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = (uint8_t)ColourIndex::PlayerMissile0; + } + + if(has_playfield || has_ball) + { + colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayfieldBall; + } + } } void TIA::set_output_mode(Atari2600::TIA::OutputMode output_mode) @@ -131,7 +191,7 @@ int TIA::get_cycles_until_horizontal_blank(unsigned int from_offset) void TIA::set_background_colour(uint8_t colour) { - background_colour_ = colour; + colour_palette_[(int)ColourIndex::Background] = colour; } void TIA::set_playfield(uint16_t offset, uint8_t value) @@ -156,13 +216,24 @@ void TIA::set_playfield(uint16_t offset, uint8_t value) void TIA::set_playfield_control_and_ball_size(uint8_t value) { background_half_mask_ = value & 1; - playfield_is_above_players_ = !!(value & 4); - playfield_is_in_score_mode_ = !playfield_is_above_players_ && (value & 2); + switch(value & 6) + { + case 0: + playfield_priority_ = PlayfieldPriority::Standard; + break; + case 2: + playfield_priority_ = PlayfieldPriority::Score; + break; + case 4: + case 6: + playfield_priority_ = PlayfieldPriority::OnTop; + break; + } } void TIA::set_playfield_ball_colour(uint8_t colour) { - playfield_ball_colour_ = colour; + colour_palette_[(int)ColourIndex::PlayfieldBall] = colour; } void TIA::set_player_number_and_size(int player, uint8_t value) @@ -207,7 +278,7 @@ void TIA::set_player_delay(int player, bool delay) void TIA::set_player_position(int player) { - player_[player].position = ((horizontal_counter_ - 68) + 6)%160; + player_[player].position = ((horizontal_counter_ - first_pixel_cycle) + 6)%160; } void TIA::set_player_motion(int player, uint8_t motion) @@ -217,7 +288,7 @@ void TIA::set_player_motion(int player, uint8_t motion) void TIA::set_player_missile_colour(int player, uint8_t colour) { - player_[player].colour = colour; + colour_palette_[(int)ColourIndex::PlayerMissile0 + player] = colour; } void TIA::set_missile_enable(int missile, bool enabled) @@ -262,11 +333,12 @@ void TIA::clear_motion() uint8_t TIA::get_collision_flags(int offset) { - return 0x00; + return (uint8_t)((collision_flags_ >> (offset << 1)) << 6) & 0xc0; } void TIA::clear_collision_flags() { + collision_flags_ = 0; } void TIA::output_for_cycles(int number_of_cycles) @@ -323,36 +395,76 @@ void TIA::output_for_cycles(int number_of_cycles) if(output_mode_ & blank_flag) { - if(pixel_target_) - { - crt_->output_data((unsigned int)((horizontal_counter_ - pixel_target_origin_) * 2), 2); - pixel_target_ = nullptr; - } + if(pixels_start_location_) output_pixels(pixels_start_location_, output_cursor); int duration = std::min(228, horizontal_counter_) - output_cursor; crt_->output_blank((unsigned int)(duration * 2)); } else { - if(!pixel_target_) + if(!pixels_start_location_) pixels_start_location_ = output_cursor; + + // accumulate an OR'dversion of the output into the collision buffer + draw_playfield(output_cursor, horizontal_counter_); + + // accumulate collision flags + while(output_cursor < horizontal_counter_) { - pixel_target_ = crt_->allocate_write_area((unsigned int)(228 - output_cursor)); - pixel_target_origin_ = output_cursor; + uint8_t buffer_value = collision_buffer_[output_cursor - first_pixel_cycle]; + collision_flags_ |= collision_flags_by_buffer_vaules_[buffer_value]; + output_cursor++; } - if(pixel_target_) - { - draw_background(pixel_target_, output_cursor, horizontal_counter_); - output_cursor = horizontal_counter_; - } else output_cursor = horizontal_counter_; + if(horizontal_counter_ == cycles_per_line) { - crt_->output_data((unsigned int)((horizontal_counter_ - pixel_target_origin_) * 2), 2); - pixel_target_ = nullptr; + output_pixels(pixels_start_location_, cycles_per_line); } } horizontal_counter_ %= cycles_per_line; } +void TIA::output_pixels(int start, int end) +{ + // seek a buffer and convert to pixels + uint8_t *pixel_target = crt_->allocate_write_area((unsigned int)(end - start)); + + if(pixel_target) + { + if(playfield_priority_ == PlayfieldPriority::Score) + { + while(start < end && start < first_pixel_cycle + 80) + { + uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; + *pixel_target = colour_palette_[colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][buffer_value]]; + start++; + pixel_target++; + } + while(start < end) + { + uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; + *pixel_target = colour_palette_[colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][buffer_value]]; + start++; + pixel_target++; + } + } + else + { + int table_index = (int)((playfield_priority_ == PlayfieldPriority::Standard) ? ColourMode::Standard : ColourMode::OnTop); + while(start < end) + { + uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; + *pixel_target = colour_palette_[colour_mask_by_mode_collision_flags_[table_index][buffer_value]]; + start++; + pixel_target++; + } + } + } + + crt_->output_data((unsigned int)((end - start) * 2), 2); + + pixels_start_location_ = 0; +} + void TIA::output_line() { switch(output_mode_) @@ -378,18 +490,13 @@ void TIA::output_line() #pragma mark - Background and playfield -void TIA::draw_background(uint8_t *target, int start, int length) const +void TIA::draw_playfield(int start, int end) { - if(!target) return; int position = start; - while(length--) + while(position < end) { - int offset = (position - 68) >> 2; - target[position - pixel_target_origin_] = ((background_[(offset/20)&background_half_mask_] >> (offset%20))&1) ? playfield_ball_colour_ : background_colour_; + int offset = (position - first_pixel_cycle) >> 2; + collision_buffer_[position - first_pixel_cycle] = (background_[(offset/20)&background_half_mask_] >> (offset%20))&1; position++; } } - -void TIA::draw_playfield(uint8_t *target, int start, int length) const -{ -} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index b9e56379e..366b5e4c1 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -75,8 +75,10 @@ class TIA { inline void output_for_cycles(int number_of_cycles); inline void output_line(); - inline void draw_background(uint8_t *target, int start, int length) const; - inline void draw_playfield(uint8_t *target, int start, int length) const; + inline void draw_playfield(int start, int end); + + int pixels_start_location_; + inline void output_pixels(int start, int end); // the master counter; counts from 0 to 228 with all visible pixels being in the final 160 int horizontal_counter_; @@ -85,24 +87,43 @@ class TIA { int output_mode_; // keeps track of the target pixel buffer for this line and when it was acquired, and a corresponding collision buffer - uint8_t *pixel_target_; - int pixel_target_origin_; uint8_t collision_buffer_[160]; enum class CollisionType : uint8_t { - Playfield, - Sprite1, - Sprite2, - Missile1, - Missile2, - Ball + Playfield = (1 << 0), + Ball = (1 << 1), + Player0 = (1 << 2), + Player1 = (1 << 3), + Missile0 = (1 << 4), + Missile1 = (1 << 5) }; + int collision_flags_; + int collision_flags_by_buffer_vaules_[64]; + + // colour mapping tables + enum class ColourMode { + Standard = 0, + ScoreLeft, + ScoreRight, + OnTop + }; + uint8_t colour_mask_by_mode_collision_flags_[4][64]; // maps from [ColourMode][CollisionMark] to colour_pallete_ entry + + enum class ColourIndex { + Background = 0, + PlayfieldBall, + PlayerMissile1, + PlayerMissile0 + }; + uint8_t colour_palette_[4]; + // playfield state - uint8_t playfield_ball_colour_; - uint8_t background_colour_; int background_half_mask_; - bool playfield_is_in_score_mode_; - bool playfield_is_above_players_; + enum class PlayfieldPriority { + Standard, + Score, + OnTop + } playfield_priority_; uint32_t background_[2]; // contains two 20-bit bitfields representing the background state; // at index 0 is the left-hand side of the playfield with bit 0 being // the first bit to display, bit 1 the second, etc. Index 1 contains @@ -117,7 +138,6 @@ class TIA { int size; // 0 = normal, 1 = double, 2 = quad int copy_flags; // a bit field, corresponding to the first few values of NUSIZ uint8_t graphic; // the player graphic - uint8_t colour; // the player colour int reverse_mask; // 7 for a reflected player, 0 for normal uint8_t motion; // low four bits used uint8_t position; // in the range [0, 160) to indicate offset from the left margin, i.e. phase difference From 8b8eb787df7528d9acfaba06a35bf7f57cf680f2 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 6 Feb 2017 18:42:58 -0500 Subject: [PATCH 20/91] Fixed complete invisibility. --- Machines/Atari2600/TIA.cpp | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 7912efae8..1fa822a06 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -52,21 +52,26 @@ TIA::TIA() : bool has_missile0 = c & (int)(CollisionType::Missile0); bool has_missile1 = c & (int)(CollisionType::Missile1); + // all priority modes show the background if nothing else is present colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::Background; + // test 1 for standard priority: if there is a playfield or ball pixel, plot that colour if(has_playfield || has_ball) { colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = (uint8_t)ColourIndex::PlayfieldBall; } + + // test 1 for score mode: if there is a ball pixel, plot that colour if(has_ball) { colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = (uint8_t)ColourIndex::PlayfieldBall; } + // test 1 for on-top mode, test 2 for everbody else: if there is a player 1 or missile 1 pixel, plot that colour if(has_player1 || has_missile1) { colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = @@ -74,11 +79,14 @@ TIA::TIA() : colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayerMissile1; } + + // in the right-hand side of score mode, the playfield has the same priority as player 1 if(has_playfield) { colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = (uint8_t)ColourIndex::PlayerMissile1; } + // next test for everybody: if there is a player 0 or missile 0 pixel, plot that colour instead if(has_player0 || has_missile0) { colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = @@ -86,11 +94,14 @@ TIA::TIA() : colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][c] = colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayerMissile0; } + + // if this is the left-hand side of score mode, the playfield has the same priority as player 0 if(has_playfield) { colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = (uint8_t)ColourIndex::PlayerMissile0; } + // a final test for 'on top' priority mode: if the playfield or ball are visible, prefer that colour to all others if(has_playfield || has_ball) { colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayfieldBall; @@ -409,8 +420,7 @@ void TIA::output_for_cycles(int number_of_cycles) // accumulate collision flags while(output_cursor < horizontal_counter_) { - uint8_t buffer_value = collision_buffer_[output_cursor - first_pixel_cycle]; - collision_flags_ |= collision_flags_by_buffer_vaules_[buffer_value]; + collision_flags_ |= collision_flags_by_buffer_vaules_[collision_buffer_[output_cursor - first_pixel_cycle]]; output_cursor++; } @@ -426,7 +436,8 @@ void TIA::output_for_cycles(int number_of_cycles) void TIA::output_pixels(int start, int end) { // seek a buffer and convert to pixels - uint8_t *pixel_target = crt_->allocate_write_area((unsigned int)(end - start)); + const unsigned int length = (unsigned int)(end - start); + uint8_t *pixel_target = crt_->allocate_write_area(length); if(pixel_target) { @@ -460,7 +471,7 @@ void TIA::output_pixels(int start, int end) } } - crt_->output_data((unsigned int)((end - start) * 2), 2); + crt_->output_data(length * 2, 2); pixels_start_location_ = 0; } From 474e2e8d2c8e464f3e3db8169301464367ae025e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 6 Feb 2017 20:09:12 -0500 Subject: [PATCH 21/91] Fixed once again to respect mid-line palette changes. --- Machines/Atari2600/TIA.cpp | 77 +++++++++++++++++++++----------------- Machines/Atari2600/TIA.hpp | 1 + 2 files changed, 43 insertions(+), 35 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 1fa822a06..0b46e37b4 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -23,6 +23,7 @@ TIA::TIA() : horizontal_counter_(0), pixels_start_location_(0), output_mode_(0), + pixel_target_(nullptr), background_{0, 0}, background_half_mask_(0) { @@ -406,17 +407,30 @@ void TIA::output_for_cycles(int number_of_cycles) if(output_mode_ & blank_flag) { - if(pixels_start_location_) output_pixels(pixels_start_location_, output_cursor); + if(pixel_target_) + { + output_pixels(pixels_start_location_, output_cursor); + crt_->output_data((unsigned int)(output_cursor - pixels_start_location_) * 2, 2); + pixel_target_ = nullptr; + pixels_start_location_ = 0; + } int duration = std::min(228, horizontal_counter_) - output_cursor; crt_->output_blank((unsigned int)(duration * 2)); } else { - if(!pixels_start_location_) pixels_start_location_ = output_cursor; + if(!pixels_start_location_) + { + pixels_start_location_ = output_cursor; + pixel_target_ = crt_->allocate_write_area(160); + } // accumulate an OR'dversion of the output into the collision buffer draw_playfield(output_cursor, horizontal_counter_); + // convert that into pixels + if(pixel_target_) output_pixels(output_cursor, horizontal_counter_); + // accumulate collision flags while(output_cursor < horizontal_counter_) { @@ -426,7 +440,9 @@ void TIA::output_for_cycles(int number_of_cycles) if(horizontal_counter_ == cycles_per_line) { - output_pixels(pixels_start_location_, cycles_per_line); + crt_->output_data((unsigned int)(output_cursor - pixels_start_location_) * 2, 2); + pixel_target_ = nullptr; + pixels_start_location_ = 0; } } @@ -435,45 +451,36 @@ void TIA::output_for_cycles(int number_of_cycles) void TIA::output_pixels(int start, int end) { - // seek a buffer and convert to pixels - const unsigned int length = (unsigned int)(end - start); - uint8_t *pixel_target = crt_->allocate_write_area(length); + int target_position = start - pixels_start_location_; - if(pixel_target) + if(playfield_priority_ == PlayfieldPriority::Score) { - if(playfield_priority_ == PlayfieldPriority::Score) + while(start < end && start < first_pixel_cycle + 80) { - while(start < end && start < first_pixel_cycle + 80) - { - uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; - *pixel_target = colour_palette_[colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][buffer_value]]; - start++; - pixel_target++; - } - while(start < end) - { - uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; - *pixel_target = colour_palette_[colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][buffer_value]]; - start++; - pixel_target++; - } + uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; + pixel_target_[target_position] = colour_palette_[colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][buffer_value]]; + start++; + target_position++; } - else + while(start < end) { - int table_index = (int)((playfield_priority_ == PlayfieldPriority::Standard) ? ColourMode::Standard : ColourMode::OnTop); - while(start < end) - { - uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; - *pixel_target = colour_palette_[colour_mask_by_mode_collision_flags_[table_index][buffer_value]]; - start++; - pixel_target++; - } + uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; + pixel_target_[target_position] = colour_palette_[colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreRight][buffer_value]]; + start++; + target_position++; + } + } + else + { + int table_index = (int)((playfield_priority_ == PlayfieldPriority::Standard) ? ColourMode::Standard : ColourMode::OnTop); + while(start < end) + { + uint8_t buffer_value = collision_buffer_[start - first_pixel_cycle]; + pixel_target_[target_position] = colour_palette_[colour_mask_by_mode_collision_flags_[table_index][buffer_value]]; + start++; + target_position++; } } - - crt_->output_data(length * 2, 2); - - pixels_start_location_ = 0; } void TIA::output_line() diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 366b5e4c1..d6c18f2f4 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -78,6 +78,7 @@ class TIA { inline void draw_playfield(int start, int end); int pixels_start_location_; + uint8_t *pixel_target_; inline void output_pixels(int start, int end); // the master counter; counts from 0 to 228 with all visible pixels being in the final 160 From ba165bb70a700aeae7f4172f405428dad6532bf7 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 6 Feb 2017 21:15:55 -0500 Subject: [PATCH 22/91] Made an attempt properly to populate collision registers from the collision buffer. --- Machines/Atari2600/TIA.cpp | 24 +++++++++++++++++++----- 1 file changed, 19 insertions(+), 5 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 0b46e37b4..6e84bc5d3 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -39,11 +39,6 @@ TIA::TIA() : ); } - for(int c = 0; c < 64; c++) - { - collision_flags_by_buffer_vaules_[c] = 0; // TODO - } - for(int c = 0; c < 64; c++) { bool has_playfield = c & (int)(CollisionType::Playfield); @@ -53,6 +48,25 @@ TIA::TIA() : bool has_missile0 = c & (int)(CollisionType::Missile0); bool has_missile1 = c & (int)(CollisionType::Missile1); + uint8_t collision_registers[8]; + collision_registers[0] = ((has_missile0 && has_player1) ? 0x80 : 0x00) | ((has_missile0 && has_player0) ? 0x40 : 0x00); + collision_registers[1] = ((has_missile1 && has_player0) ? 0x80 : 0x00) | ((has_missile1 && has_player1) ? 0x40 : 0x00); + collision_registers[2] = ((has_playfield && has_player0) ? 0x80 : 0x00) | ((has_ball && has_player0) ? 0x40 : 0x00); + collision_registers[3] = ((has_playfield && has_player1) ? 0x80 : 0x00) | ((has_ball && has_player1) ? 0x40 : 0x00); + collision_registers[4] = ((has_playfield && has_missile0) ? 0x80 : 0x00) | ((has_ball && has_missile0) ? 0x40 : 0x00); + collision_registers[5] = ((has_playfield && has_missile1) ? 0x80 : 0x00) | ((has_ball && has_missile1) ? 0x40 : 0x00); + collision_registers[6] = ((has_playfield && has_ball) ? 0x80 : 0x00); + collision_registers[7] = ((has_player0 && has_player1) ? 0x80 : 0x00) | ((has_missile0 && has_missile1) ? 0x40 : 0x00); + collision_flags_by_buffer_vaules_[c] = + (collision_registers[0] >> 6) | + (collision_registers[1] >> 4) | + (collision_registers[2] >> 2) | + (collision_registers[3] >> 0) | + (collision_registers[4] << 2) | + (collision_registers[5] << 4) | + (collision_registers[6] << 6) | + (collision_registers[7] << 8); + // all priority modes show the background if nothing else is present colour_mask_by_mode_collision_flags_[(int)ColourMode::Standard][c] = colour_mask_by_mode_collision_flags_[(int)ColourMode::ScoreLeft][c] = From 8f5039130ce42f060d950e83472170056f1cf87e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 6 Feb 2017 21:48:41 -0500 Subject: [PATCH 23/91] Changed index naming order to ensure no out-of-bounds accesses. --- Machines/Atari2600/TIA.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index d6c18f2f4..599585ddb 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -113,8 +113,8 @@ class TIA { enum class ColourIndex { Background = 0, PlayfieldBall, - PlayerMissile1, - PlayerMissile0 + PlayerMissile0, + PlayerMissile1 }; uint8_t colour_palette_[4]; From 944d835eea610fd794fe7513a632f624da8591e0 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 6 Feb 2017 21:59:28 -0500 Subject: [PATCH 24/91] Switched explicitly to an accumulation model for filling the collision buffer. --- Machines/Atari2600/TIA.cpp | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 6e84bc5d3..e2c9efc10 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -386,6 +386,11 @@ void TIA::output_for_cycles(int number_of_cycles) int output_cursor = horizontal_counter_; horizontal_counter_ += number_of_cycles; + if(!output_cursor) + { + memset(collision_buffer_, 0, sizeof(collision_buffer_)); + } + #define Period(function, target) \ if(output_cursor < target) \ { \ @@ -419,6 +424,8 @@ void TIA::output_for_cycles(int number_of_cycles) break; } +#undef Period + if(output_mode_ & blank_flag) { if(pixel_target_) @@ -528,7 +535,7 @@ void TIA::draw_playfield(int start, int end) while(position < end) { int offset = (position - first_pixel_cycle) >> 2; - collision_buffer_[position - first_pixel_cycle] = (background_[(offset/20)&background_half_mask_] >> (offset%20))&1; + collision_buffer_[position - first_pixel_cycle] |= (background_[(offset/20)&background_half_mask_] >> (offset%20))&1; position++; } } From a4774997249f6198d41bd5e636697f9cd493c1db Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 7 Feb 2017 22:14:45 -0500 Subject: [PATCH 25/91] Got a bit more explicit with range returned by `get_cycles_until_horizontal_blank` and hence attempted a more thorough (/correct) version of WSYNC. --- Machines/Atari2600/Atari2600.cpp | 11 ++++++----- Machines/Atari2600/TIA.cpp | 2 +- Machines/Atari2600/TIA.hpp | 4 ++++ 3 files changed, 11 insertions(+), 6 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index 7f7f70ea6..26b1a6c7d 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -54,14 +54,15 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin // leap to the end of ready only once ready is signalled — because on a 6502 ready doesn't take // effect until the next read; therefore it isn't safe to assume that signalling ready immediately // skips to the end of the line. - if(operation == CPU6502::BusOperation::Ready) { + if(operation == CPU6502::BusOperation::Ready) cycles_run_for = (unsigned int)tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_); - set_ready_line(false); - } cycles_since_speaker_update_ += cycles_run_for; cycles_since_video_update_ += cycles_run_for; + if(!tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_)) + set_ready_line(false); + if(operation != CPU6502::BusOperation::Ready) { // check for a paging access @@ -134,8 +135,8 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin case 0x01: update_video(); tia_->set_blank(!!(*value & 0x02)); break; case 0x02: - set_ready_line(true); - // TODO: if(horizontal_timer_) + if(tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_)) + set_ready_line(true); break; case 0x03: update_video(); tia_->reset_horizontal_counter(); break; // TODO: audio will now be out of synchronisation — fix diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index e2c9efc10..d24b205f7 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -212,7 +212,7 @@ void TIA::reset_horizontal_counter() int TIA::get_cycles_until_horizontal_blank(unsigned int from_offset) { - return 4 + (cycles_per_line - (horizontal_counter_ + (int)from_offset) % cycles_per_line); + return (cycles_per_line - (horizontal_counter_ + (int)from_offset) % cycles_per_line) % cycles_per_line; } void TIA::set_background_colour(uint8_t colour) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 599585ddb..ceda0c412 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -34,6 +34,10 @@ class TIA { void set_blank(bool blank); void reset_horizontal_counter(); // Reset is delayed by four cycles. + /*! + @returns the number of cycles between (current TIA time) + from_offset to the current or + next horizontal blanking period. Returns numbers in the range [0, 227]. + */ int get_cycles_until_horizontal_blank(unsigned int from_offset); void set_background_colour(uint8_t colour); From dcb75840608442ee91563d0035a85b61b9f23a83 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 8 Feb 2017 07:30:32 -0500 Subject: [PATCH 26/91] Added the four-cycle playfield output latency and ensured you can't get smaller-than-usual pixels by rapid register value changing. --- Machines/Atari2600/TIA.cpp | 33 ++++++++++++++++++++++++--------- 1 file changed, 24 insertions(+), 9 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index d24b205f7..e8692b24d 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -391,6 +391,11 @@ void TIA::output_for_cycles(int number_of_cycles) memset(collision_buffer_, 0, sizeof(collision_buffer_)); } + // accumulate an OR'dversion of the output into the collision buffer + draw_playfield(output_cursor, horizontal_counter_); + + // convert to television signals + #define Period(function, target) \ if(output_cursor < target) \ { \ @@ -446,9 +451,6 @@ void TIA::output_for_cycles(int number_of_cycles) pixel_target_ = crt_->allocate_write_area(160); } - // accumulate an OR'dversion of the output into the collision buffer - draw_playfield(output_cursor, horizontal_counter_); - // convert that into pixels if(pixel_target_) output_pixels(output_cursor, horizontal_counter_); @@ -527,15 +529,28 @@ void TIA::output_line() } } -#pragma mark - Background and playfield +#pragma mark - Playfield output void TIA::draw_playfield(int start, int end) { - int position = start; - while(position < end) + // don't do anything if this window ends too early + if(end < first_pixel_cycle - 4) return; + + // look at what needs to be output four cycles into the future, to model playfield output latency + start += 4; + end += 4; + + // clip to drawable bounds + start = std::max(start, first_pixel_cycle); + end = std::min(end, 228); + + // proceed along four-pixel boundaries, plotting four pixels at a time + int aligned_position = (start + 3)&~3; + while(aligned_position < end) { - int offset = (position - first_pixel_cycle) >> 2; - collision_buffer_[position - first_pixel_cycle] |= (background_[(offset/20)&background_half_mask_] >> (offset%20))&1; - position++; + int offset = (aligned_position - first_pixel_cycle) >> 2; + uint32_t value = ((background_[(offset/20)&background_half_mask_] >> (offset%20))&1) * 0x01010101; + *(uint32_t *)&collision_buffer_[aligned_position - first_pixel_cycle] |= value; + aligned_position += 4; } } From 70745286a5491fa39efaf718551b74cf06d93863 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 8 Feb 2017 20:25:23 -0500 Subject: [PATCH 27/91] Ensured this array is properly aligned for the `uin32_t` accesses I intend to make for background drawing. --- Machines/Atari2600/TIA.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index ceda0c412..e25ccddf5 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -92,7 +92,7 @@ class TIA { int output_mode_; // keeps track of the target pixel buffer for this line and when it was acquired, and a corresponding collision buffer - uint8_t collision_buffer_[160]; + alignas(alignof(uint32_t)) uint8_t collision_buffer_[160]; enum class CollisionType : uint8_t { Playfield = (1 << 0), Ball = (1 << 1), From 2e9ef2b0ef7e4b7fc358f3b0d51874aee09148fe Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 9 Feb 2017 18:37:19 -0500 Subject: [PATCH 28/91] Took a shot at reinstating the horizontal blank extend flag. --- Machines/Atari2600/TIA.cpp | 16 +++++++++++++++- Machines/Atari2600/TIA.hpp | 3 +++ 2 files changed, 18 insertions(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index e8692b24d..f999285db 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -351,6 +351,7 @@ void TIA::set_ball_motion(uint8_t motion) void TIA::move() { + horizontal_blank_extend_ = true; } void TIA::clear_motion() @@ -389,9 +390,10 @@ void TIA::output_for_cycles(int number_of_cycles) if(!output_cursor) { memset(collision_buffer_, 0, sizeof(collision_buffer_)); + horizontal_blank_extend_ = false; } - // accumulate an OR'dversion of the output into the collision buffer + // accumulate an OR'd version of the output into the collision buffer draw_playfield(output_cursor, horizontal_counter_); // convert to television signals @@ -476,6 +478,16 @@ void TIA::output_pixels(int start, int end) { int target_position = start - pixels_start_location_; + if(start < first_pixel_cycle+8 && horizontal_blank_extend_) + { + while(start < end && start < first_pixel_cycle+8) + { + pixel_target_[target_position] = 0; + start++; + target_position++; + } + } + if(playfield_priority_ == PlayfieldPriority::Score) { while(start < end && start < first_pixel_cycle + 80) @@ -519,12 +531,14 @@ void TIA::output_line() crt_->output_sync(32); crt_->output_blank(32); crt_->output_sync(392); + horizontal_blank_extend_ = false; break; case blank_flag: crt_->output_blank(32); crt_->output_sync(32); crt_->output_default_colour_burst(32); crt_->output_blank(360); + horizontal_blank_extend_ = false; break; } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index e25ccddf5..a7c310475 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -153,6 +153,9 @@ class TIA { struct Missile { int size; // 0 = 1 pixel, 1 = 2 pixels, etc } missile_[2]; + + // movement + bool horizontal_blank_extend_; }; } From 3b20d862f0befc74cd184cbdf346f13c9c63c05c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 9 Feb 2017 20:53:42 -0500 Subject: [PATCH 29/91] Made an initial attempt to mark sprite positions. But without hmove implemented, they're all over the place. --- Machines/Atari2600/TIA.cpp | 31 ++++++++++++++++++++++++++++++- Machines/Atari2600/TIA.hpp | 21 +++++++++++---------- 2 files changed, 41 insertions(+), 11 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index f999285db..0094a1dfc 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -304,7 +304,7 @@ void TIA::set_player_delay(int player, bool delay) void TIA::set_player_position(int player) { - player_[player].position = ((horizontal_counter_ - first_pixel_cycle) + 6)%160; + player_[player].position = 0; } void TIA::set_player_motion(int player, uint8_t motion) @@ -395,6 +395,8 @@ void TIA::output_for_cycles(int number_of_cycles) // accumulate an OR'd version of the output into the collision buffer draw_playfield(output_cursor, horizontal_counter_); + draw_player(player_[0], CollisionType::Player0, output_cursor, horizontal_counter_); + draw_player(player_[1], CollisionType::Player1, output_cursor, horizontal_counter_); // convert to television signals @@ -568,3 +570,30 @@ void TIA::draw_playfield(int start, int end) aligned_position += 4; } } + +#pragma mark - Player output + +void TIA::draw_player(Player &player, CollisionType identity, int start, int end) +{ + // don't do anything if this window ends too early + if(end < first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0)) return; + + int length = end - start; + + // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy + if(player.position + length >= 160) + { + int trigger_position = 160 - player.position; + + int terminus = std::min(160, trigger_position+8); + while(trigger_position < terminus) + { + collision_buffer_[trigger_position] |= (uint8_t)identity; + trigger_position++; + } + } + + // update position counter + player.position = (player.position + end - start) % 160; + +} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index a7c310475..db1850d58 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -75,16 +75,6 @@ class TIA { private: std::shared_ptr crt_; - // drawing methods - inline void output_for_cycles(int number_of_cycles); - inline void output_line(); - - inline void draw_playfield(int start, int end); - - int pixels_start_location_; - uint8_t *pixel_target_; - inline void output_pixels(int start, int end); - // the master counter; counts from 0 to 228 with all visible pixels being in the final 160 int horizontal_counter_; @@ -156,6 +146,17 @@ class TIA { // movement bool horizontal_blank_extend_; + + // drawing methods and state + inline void output_for_cycles(int number_of_cycles); + inline void output_line(); + + inline void draw_playfield(int start, int end); + inline void draw_player(Player &player, CollisionType identity, int start, int end); + + int pixels_start_location_; + uint8_t *pixel_target_; + inline void output_pixels(int start, int end); }; } From 64d5712d1d70dcf7a7da2ef5d7b1e0940bae16a5 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 10 Feb 2017 07:23:43 -0500 Subject: [PATCH 30/91] Added an incorrectly-coded version of horizontal move, at least so that I can verify that information is going into the correct slots. --- Machines/Atari2600/TIA.cpp | 37 ++++++++++++++++++++++++++----------- Machines/Atari2600/TIA.hpp | 17 +++++++++++++---- 2 files changed, 39 insertions(+), 15 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 0094a1dfc..ba566a39d 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -304,12 +304,12 @@ void TIA::set_player_delay(int player, bool delay) void TIA::set_player_position(int player) { - player_[player].position = 0; + position_[(int)MotionIndex::Player0 + player] = 0; } void TIA::set_player_motion(int player, uint8_t motion) { - player_[player].motion = motion >> 4; + motion_[(int)MotionIndex::Player0 + player] = motion >> 4; } void TIA::set_player_missile_colour(int player, uint8_t colour) @@ -352,10 +352,12 @@ void TIA::set_ball_motion(uint8_t motion) void TIA::move() { horizontal_blank_extend_ = true; + is_moving_[0] = is_moving_[1] = is_moving_[2] = is_moving_[3] = is_moving_[4] = true; } void TIA::clear_motion() { + motion_[0] = motion_[1] = motion_[2] = motion_[3] = motion_[4] = 0; } uint8_t TIA::get_collision_flags(int offset) @@ -395,8 +397,8 @@ void TIA::output_for_cycles(int number_of_cycles) // accumulate an OR'd version of the output into the collision buffer draw_playfield(output_cursor, horizontal_counter_); - draw_player(player_[0], CollisionType::Player0, output_cursor, horizontal_counter_); - draw_player(player_[1], CollisionType::Player1, output_cursor, horizontal_counter_); + draw_player(player_[0], CollisionType::Player0, (int)MotionIndex::Player0, output_cursor, horizontal_counter_); + draw_player(player_[1], CollisionType::Player1, (int)MotionIndex::Player1, output_cursor, horizontal_counter_); // convert to television signals @@ -573,27 +575,40 @@ void TIA::draw_playfield(int start, int end) #pragma mark - Player output -void TIA::draw_player(Player &player, CollisionType identity, int start, int end) +void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) { // don't do anything if this window ends too early - if(end < first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0)) return; + int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); + if(end < first_pixel) return; + if(start < first_pixel) start = first_pixel; int length = end - start; + uint8_t &position = position_[position_identity]; + + // quick hack! + if(is_moving_[position_identity]) + { + int motion = motion_[position_identity]; + position += 8; + position += motion; + if(motion&8) position -= 16; + is_moving_[position_identity] = false; + position = (position + 160)%160; + } // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy - if(player.position + length >= 160) + if(player.graphic && position + length >= 160) { - int trigger_position = 160 - player.position; + int trigger_position = 160 - position; int terminus = std::min(160, trigger_position+8); while(trigger_position < terminus) { - collision_buffer_[trigger_position] |= (uint8_t)identity; + collision_buffer_[trigger_position] |= (uint8_t)collision_identity; trigger_position++; } } // update position counter - player.position = (player.position + end - start) % 160; - + position = (position + end - start) % 160; } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index db1850d58..ac704c629 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -134,9 +134,6 @@ class TIA { int copy_flags; // a bit field, corresponding to the first few values of NUSIZ uint8_t graphic; // the player graphic int reverse_mask; // 7 for a reflected player, 0 for normal - uint8_t motion; // low four bits used - uint8_t position; // in the range [0, 160) to indicate offset from the left margin, i.e. phase difference - // between the player counter and the background pixel counter. } player_[2]; // missile state @@ -146,13 +143,25 @@ class TIA { // movement bool horizontal_blank_extend_; + uint8_t motion_[5]; + uint8_t position_[5]; + bool is_moving_[5]; + enum class MotionIndex : uint8_t { + Ball, + Player0, + Player1, + Missile0, + Missile1 + }; // drawing methods and state inline void output_for_cycles(int number_of_cycles); inline void output_line(); inline void draw_playfield(int start, int end); - inline void draw_player(Player &player, CollisionType identity, int start, int end); + inline void draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); + + inline void update_motion(int start, int end); int pixels_start_location_; uint8_t *pixel_target_; From 40d3f5f7f6874549ae299dc46916a419870bbd70 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 08:26:09 -0500 Subject: [PATCH 31/91] Attempted properly to respect `start`. --- Machines/Atari2600/TIA.cpp | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index ba566a39d..1fa3b28b4 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -588,10 +588,8 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // quick hack! if(is_moving_[position_identity]) { - int motion = motion_[position_identity]; - position += 8; + int motion = motion_[position_identity] ^ 8; position += motion; - if(motion&8) position -= 16; is_moving_[position_identity] = false; position = (position + 160)%160; } @@ -599,7 +597,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy if(player.graphic && position + length >= 160) { - int trigger_position = 160 - position; + int trigger_position = 160 - position + 6 + start - first_pixel_cycle; int terminus = std::min(160, trigger_position+8); while(trigger_position < terminus) From 327c19a222577e55b9aa98c5c3c71778fa75a4ee Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 12:58:47 -0500 Subject: [PATCH 32/91] Slightly shuffled to avoid a race condition on the best-effort updater. --- .../Documents/MachineDocument.swift | 24 +++++++++---------- .../Mac/Clock Signal/Views/CSOpenGLView.h | 2 +- 2 files changed, 12 insertions(+), 14 deletions(-) diff --git a/OSBindings/Mac/Clock Signal/Documents/MachineDocument.swift b/OSBindings/Mac/Clock Signal/Documents/MachineDocument.swift index 0d946e3c2..995e35027 100644 --- a/OSBindings/Mac/Clock Signal/Documents/MachineDocument.swift +++ b/OSBindings/Mac/Clock Signal/Documents/MachineDocument.swift @@ -31,24 +31,14 @@ class MachineDocument: return NSSize(width: 4.0, height: 3.0) } - @IBOutlet weak var openGLView: CSOpenGLView! { - didSet { - openGLView.delegate = self - openGLView.responderDelegate = self - } - } - + @IBOutlet weak var openGLView: CSOpenGLView! @IBOutlet var optionsPanel: MachinePanel! @IBAction func showOptions(_ sender: AnyObject!) { optionsPanel?.setIsVisible(true) } fileprivate var audioQueue: CSAudioQueue! = nil - fileprivate lazy var bestEffortUpdater: CSBestEffortUpdater = { - let updater = CSBestEffortUpdater() - updater.delegate = self - return updater - }() + fileprivate var bestEffortUpdater: CSBestEffortUpdater! override var windowNibName: String? { return "MachineDocument" @@ -64,8 +54,16 @@ class MachineDocument: self.machine.setView(self.openGLView, aspectRatio: Float(displayAspectRatio.width / displayAspectRatio.height)) }) - setupClockRate() self.machine.delegate = self + self.bestEffortUpdater = CSBestEffortUpdater() + self.bestEffortUpdater.delegate = self + + // callbacks from the OpenGL may come on a different thread, immediately following the .delegate set; + // hence the full setup of the best-effort updater prior to setting self as a delegate + self.openGLView.delegate = self + self.openGLView.responderDelegate = self + + setupClockRate() self.optionsPanel?.establishStoredOptions() // bring OpenGL view-holding window on top of the options panel diff --git a/OSBindings/Mac/Clock Signal/Views/CSOpenGLView.h b/OSBindings/Mac/Clock Signal/Views/CSOpenGLView.h index 9ecae793d..34d08584e 100644 --- a/OSBindings/Mac/Clock Signal/Views/CSOpenGLView.h +++ b/OSBindings/Mac/Clock Signal/Views/CSOpenGLView.h @@ -53,7 +53,7 @@ */ @interface CSOpenGLView : NSOpenGLView -@property (nonatomic, weak, nullable) id delegate; +@property (atomic, weak, nullable) id delegate; @property (nonatomic, weak, nullable) id responderDelegate; /*! From 8de6caf6ff2296f383ef4f625eee967ea7232947 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 12:59:13 -0500 Subject: [PATCH 33/91] Started trying to get into a proper structure here. Chickened out. --- Machines/Atari2600/TIA.cpp | 45 +++++++++++++++++++++----------------- Machines/Atari2600/TIA.hpp | 1 + 2 files changed, 26 insertions(+), 20 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 1fa3b28b4..d9f34cbb2 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -353,6 +353,7 @@ void TIA::move() { horizontal_blank_extend_ = true; is_moving_[0] = is_moving_[1] = is_moving_[2] = is_moving_[3] = is_moving_[4] = true; + horizontal_move_start_time_ = horizontal_counter_; } void TIA::clear_motion() @@ -582,30 +583,34 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(end < first_pixel) return; if(start < first_pixel) start = first_pixel; - int length = end - start; uint8_t &position = position_[position_identity]; + uint8_t &motion = motion_[position_identity]; + bool &is_moving = is_moving_[position_identity]; +// while(start < end) +// { + int length = end - start; - // quick hack! - if(is_moving_[position_identity]) - { - int motion = motion_[position_identity] ^ 8; - position += motion; - is_moving_[position_identity] = false; - position = (position + 160)%160; - } - - // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy - if(player.graphic && position + length >= 160) - { - int trigger_position = 160 - position + 6 + start - first_pixel_cycle; - - int terminus = std::min(160, trigger_position+8); - while(trigger_position < terminus) + // quick hack! + if(is_moving) { - collision_buffer_[trigger_position] |= (uint8_t)collision_identity; - trigger_position++; + position += (motion ^ 8); + is_moving_[position_identity] = false; + position = (position + 160)%160; } - } + + // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy + if(player.graphic && position + length >= 160) + { + int trigger_position = 160 - position + 5 + start - first_pixel_cycle; + + int terminus = std::min(160, trigger_position+8); + while(trigger_position < terminus) + { + collision_buffer_[trigger_position] |= (uint8_t)collision_identity; + trigger_position++; + } + } +// } // update position counter position = (position + end - start) % 160; diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index ac704c629..789d786d0 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -143,6 +143,7 @@ class TIA { // movement bool horizontal_blank_extend_; + int horizontal_move_start_time_; uint8_t motion_[5]; uint8_t position_[5]; bool is_moving_[5]; From 905ed1f87bcbee4799133e0e600849d1e338f015 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 13:16:53 -0500 Subject: [PATCH 34/91] Switched to the more natural type, which is also signed, making my logic less prone to error. --- Machines/Atari2600/TIA.cpp | 4 ++-- Machines/Atari2600/TIA.hpp | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index d9f34cbb2..b46ccbc29 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -583,8 +583,8 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(end < first_pixel) return; if(start < first_pixel) start = first_pixel; - uint8_t &position = position_[position_identity]; - uint8_t &motion = motion_[position_identity]; + int &position = position_[position_identity]; + int &motion = motion_[position_identity]; bool &is_moving = is_moving_[position_identity]; // while(start < end) // { diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 789d786d0..28140b4b9 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -144,8 +144,8 @@ class TIA { // movement bool horizontal_blank_extend_; int horizontal_move_start_time_; - uint8_t motion_[5]; - uint8_t position_[5]; + int motion_[5]; + int position_[5]; bool is_moving_[5]; enum class MotionIndex : uint8_t { Ball, From 6381e4e1b041c3e93a5f52eb59e16b1acfca73f7 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 13:34:36 -0500 Subject: [PATCH 35/91] All that's happened to position is that numbers have been added to it. So it can't be negative, given that it wasn't before. So a regular modulo will do. --- Machines/Atari2600/TIA.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index b46ccbc29..e545e279d 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -595,7 +595,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in { position += (motion ^ 8); is_moving_[position_identity] = false; - position = (position + 160)%160; + position %= 160; } // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy From aff69dbc3456396e5b4537f9be418da77e45c3b5 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 13:35:22 -0500 Subject: [PATCH 36/91] =?UTF-8?q?Resolved=20spurious=20static=20analyser?= =?UTF-8?q?=20issue;=20screen=20mode=20will=20always=20be=200=E2=80=936=20?= =?UTF-8?q?but=20it=20doesn't=20know=20that.=20Setting=20a=20non-zero=20di?= =?UTF-8?q?vider=20doesn't=20feel=20worth=20worrying=20about=20for=20a=20c?= =?UTF-8?q?leaner=20compile.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Machines/Electron/Video.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/Electron/Video.cpp b/Machines/Electron/Video.cpp index 29b95b730..2a1f4688c 100644 --- a/Machines/Electron/Video.cpp +++ b/Machines/Electron/Video.cpp @@ -115,7 +115,7 @@ void VideoOutput::output_pixels(unsigned int number_of_cycles) } else { - unsigned int divider = 0; + unsigned int divider = 1; switch(screen_mode_) { case 0: case 3: divider = 2; break; From 97cdfea9e919b22d0c6eba0df73fe823f76c2b0f Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 13:36:09 -0500 Subject: [PATCH 37/91] Resolved spurious static analyser complaint: input_size and output_size aren't supposed to have defined values if input or output is null. But whatever. --- Outputs/CRT/Internals/ArrayBuilder.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Outputs/CRT/Internals/ArrayBuilder.cpp b/Outputs/CRT/Internals/ArrayBuilder.cpp index afda1c9b6..b91246d45 100644 --- a/Outputs/CRT/Internals/ArrayBuilder.cpp +++ b/Outputs/CRT/Internals/ArrayBuilder.cpp @@ -41,7 +41,7 @@ void ArrayBuilder::flush(const std::function Date: Sat, 11 Feb 2017 13:36:36 -0500 Subject: [PATCH 38/91] Ensured no attempt to call `strcmp` on `null` if a file name without an extension got into here. --- StaticAnalyser/StaticAnalyser.cpp | 70 ++++++++++++++++--------------- 1 file changed, 37 insertions(+), 33 deletions(-) diff --git a/StaticAnalyser/StaticAnalyser.cpp b/StaticAnalyser/StaticAnalyser.cpp index cd26c382c..291187996 100644 --- a/StaticAnalyser/StaticAnalyser.cpp +++ b/StaticAnalyser/StaticAnalyser.cpp @@ -84,45 +84,49 @@ std::list StaticAnalyser::GetTargets(const char *file_name) TryInsert(list, class, platforms) \ } - Format("a26", cartridges, Cartridge::BinaryDump, TargetPlatform::Atari2600) // A26 - Format("adf", disks, Disk::AcornADF, TargetPlatform::Acorn) // ADF - Format("bin", cartridges, Cartridge::BinaryDump, TargetPlatform::Atari2600) // BIN - Format("d64", disks, Disk::D64, TargetPlatform::Commodore) // D64 - Format("dsd", disks, Disk::SSD, TargetPlatform::Acorn) // DSD - Format("dsk", disks, Disk::OricMFMDSK, TargetPlatform::Oric) // DSK - Format("g64", disks, Disk::G64, TargetPlatform::Commodore) // G64 - - // PRG - if(!strcmp(lowercase_extension, "prg")) + if(lowercase_extension) { - // try instantiating as a ROM; failing that accept as a tape - try { - Insert(cartridges, Cartridge::PRG, TargetPlatform::Commodore) - } - catch(...) - { - try { - Insert(tapes, Tape::PRG, TargetPlatform::Commodore) - } catch(...) {} - } - } + Format("a26", cartridges, Cartridge::BinaryDump, TargetPlatform::Atari2600) // A26 + Format("adf", disks, Disk::AcornADF, TargetPlatform::Acorn) // ADF + Format("bin", cartridges, Cartridge::BinaryDump, TargetPlatform::Atari2600) // BIN + Format("d64", disks, Disk::D64, TargetPlatform::Commodore) // D64 + Format("dsd", disks, Disk::SSD, TargetPlatform::Acorn) // DSD + Format("dsk", disks, Disk::OricMFMDSK, TargetPlatform::Oric) // DSK + Format("g64", disks, Disk::G64, TargetPlatform::Commodore) // G64 - Format("rom", cartridges, Cartridge::BinaryDump, TargetPlatform::Acorn) // ROM - Format("ssd", disks, Disk::SSD, TargetPlatform::Acorn) // SSD - Format("tap", tapes, Tape::CommodoreTAP, TargetPlatform::Commodore) // TAP (Commodore) - Format("tap", tapes, Tape::OricTAP, TargetPlatform::Oric) // TAP (Oric) - Format("uef", tapes, Tape::UEF, TargetPlatform::Acorn) // UEF (tape) + // PRG + if(!strcmp(lowercase_extension, "prg")) + { + // try instantiating as a ROM; failing that accept as a tape + try { + Insert(cartridges, Cartridge::PRG, TargetPlatform::Commodore) + } + catch(...) + { + try { + Insert(tapes, Tape::PRG, TargetPlatform::Commodore) + } catch(...) {} + } + } + + Format("rom", cartridges, Cartridge::BinaryDump, TargetPlatform::Acorn) // ROM + Format("ssd", disks, Disk::SSD, TargetPlatform::Acorn) // SSD + Format("tap", tapes, Tape::CommodoreTAP, TargetPlatform::Commodore) // TAP (Commodore) + Format("tap", tapes, Tape::OricTAP, TargetPlatform::Oric) // TAP (Oric) + Format("uef", tapes, Tape::UEF, TargetPlatform::Acorn) // UEF (tape) #undef Format #undef Insert +#undef TryInsert - // Hand off to platform-specific determination of whether these things are actually compatible and, - // if so, how to load them. (TODO) - if(potential_platforms & (TargetPlatformType)TargetPlatform::Acorn) Acorn::AddTargets(disks, tapes, cartridges, targets); - if(potential_platforms & (TargetPlatformType)TargetPlatform::Atari2600) Atari::AddTargets(disks, tapes, cartridges, targets); - if(potential_platforms & (TargetPlatformType)TargetPlatform::Commodore) Commodore::AddTargets(disks, tapes, cartridges, targets); - if(potential_platforms & (TargetPlatformType)TargetPlatform::Oric) Oric::AddTargets(disks, tapes, cartridges, targets); + // Hand off to platform-specific determination of whether these things are actually compatible and, + // if so, how to load them. (TODO) + if(potential_platforms & (TargetPlatformType)TargetPlatform::Acorn) Acorn::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::Atari2600) Atari::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::Commodore) Commodore::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::Oric) Oric::AddTargets(disks, tapes, cartridges, targets); - free(lowercase_extension); + free(lowercase_extension); + } return targets; } From aeff59addc3dfb297091bdfcfb3a4736120a2354 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 20:25:49 -0500 Subject: [PATCH 39/91] Implemented motion 'correctly', for programs written to do all work outside of the pixel area. --- Machines/Atari2600/TIA.cpp | 42 ++++++++++++++++++++++++++++---------- 1 file changed, 31 insertions(+), 11 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index e545e279d..ba83cdfc7 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -578,25 +578,45 @@ void TIA::draw_playfield(int start, int end) void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) { - // don't do anything if this window ends too early - int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); - if(end < first_pixel) return; - if(start < first_pixel) start = first_pixel; - int &position = position_[position_identity]; int &motion = motion_[position_identity]; bool &is_moving = is_moving_[position_identity]; + + // movement is applied prior to the drawing area (as well as within) + int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); + if(is_moving) + { + // round up to the next H@1 cycle + int movement_time = (start + 3) & ~3; + while(movement_time < end && movement_time < first_pixel) + { + int movement = (movement_time - horizontal_move_start_time_) >> 2; + if(movement == (motion ^ 8)) + { + is_moving = false; + break; + } + position ++; + movement_time += 4; + } + position %= 160; + } + + // don't do any drawing if this window ends too early + if(end < first_pixel) return; + if(start < first_pixel) start = first_pixel; + // while(start < end) // { int length = end - start; // quick hack! - if(is_moving) - { - position += (motion ^ 8); - is_moving_[position_identity] = false; - position %= 160; - } +// if(is_moving) +// { +// position += (motion ^ 8); +// is_moving_[position_identity] = false; +// position %= 160; +// } // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy if(player.graphic && position + length >= 160) From b8abeced6def1b33bbbeb8193a1301e5d51e72a1 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 21:01:58 -0500 Subject: [PATCH 40/91] Made an attempt to introduce the proper eventful loop for player output. With debugging yet to occur. --- Machines/Atari2600/TIA.cpp | 95 +++++++++++++++++++++++++++----------- 1 file changed, 68 insertions(+), 27 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index ba83cdfc7..b2b860a16 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -276,7 +276,7 @@ void TIA::set_player_number_and_size(int player, uint8_t value) break; case 6: player_[player].size = 0; - player_[player].copy_flags = 7; + player_[player].copy_flags = 6; break; case 7: player_[player].size = 2; @@ -582,12 +582,12 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in int &motion = motion_[position_identity]; bool &is_moving = is_moving_[position_identity]; - // movement is applied prior to the drawing area (as well as within) + // movement works across the entire screen, so do work that falls outside of the pixel area int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); + int movement_time = (start + 3) & ~3; if(is_moving) { // round up to the next H@1 cycle - int movement_time = (start + 3) & ~3; while(movement_time < end && movement_time < first_pixel) { int movement = (movement_time - horizontal_move_start_time_) >> 2; @@ -602,36 +602,77 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in position %= 160; } - // don't do any drawing if this window ends too early + // don't continue to do any drawing if this window ends too early if(end < first_pixel) return; if(start < first_pixel) start = first_pixel; -// while(start < end) -// { - int length = end - start; - - // quick hack! -// if(is_moving) -// { -// position += (motion ^ 8); -// is_moving_[position_identity] = false; -// position %= 160; -// } - - // check for initial trigger; player.position is guaranteed to be less than 160 so this is easy - if(player.graphic && position + length >= 160) + // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion + if(is_moving || player.graphic) + { + while(start < end) { - int trigger_position = 160 - position + 5 + start - first_pixel_cycle; + int next_event_time = end; - int terminus = std::min(160, trigger_position+8); - while(trigger_position < terminus) + // is the next event a movement tick? + if(is_moving && movement_time + 4 < next_event_time) { - collision_buffer_[trigger_position] |= (uint8_t)collision_identity; - trigger_position++; + next_event_time = movement_time + 4; + } + + // is the next event a graphics draw? + int next_copy = 160; + if(player.graphic) + { + if(position < 16 && player.copy_flags&1) + { + next_copy = 16; + } else if(position < 32 && player.copy_flags&2) + { + next_copy = 32; + } else if(position < 64 && player.copy_flags&4) + { + next_copy = 64; + } + + int time_until_copy = next_copy - position; + if(start+time_until_copy < next_event_time) next_event_time = start+time_until_copy; + } + + // the next interesting event is after next_event_time cycles, so progress + position = (position + next_event_time) % 160; + start = next_event_time; + + // if the event is a motion tick, apply + if(is_moving && start == movement_time + 4) + { + int movement_step = (movement_time - horizontal_move_start_time_) >> 2; + if(movement_step == (motion ^ 8)) + { + is_moving = false; + } + else + { + position ++; + movement_time += 4; + } + } + + // if it's a draw trigger, draw (TODO: use the actual graphic) + if(position == (next_copy % 160)) + { + int cursor = start + 5 - first_pixel_cycle; + int terminus = std::min(160, cursor+8); + while(cursor < terminus) + { + collision_buffer_[cursor] |= (uint8_t)collision_identity; + cursor++; + } } } -// } - - // update position counter - position = (position + end - start) % 160; + } + else + { + // just advance the timer all in one jump + position = (position + start - end) % 160; + } } From ac444a3f34d810d4518dec5bf0ceee5f8341936b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 11 Feb 2017 21:24:14 -0500 Subject: [PATCH 41/91] Corrected both position increments and target time calculation. --- Machines/Atari2600/TIA.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index b2b860a16..b1cd3153a 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -634,12 +634,12 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in next_copy = 64; } - int time_until_copy = next_copy - position; - if(start+time_until_copy < next_event_time) next_event_time = start+time_until_copy; + int next_copy_time = start + next_copy - position; + if(next_copy_time < next_event_time) next_event_time = next_copy_time; } // the next interesting event is after next_event_time cycles, so progress - position = (position + next_event_time) % 160; + position = (position + next_event_time - start) % 160; start = next_event_time; // if the event is a motion tick, apply @@ -673,6 +673,6 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in else { // just advance the timer all in one jump - position = (position + start - end) % 160; + position = (position + end - start) % 160; } } From 40954d6a2a8e595b81e1e2211d0900098e37055b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 11:31:17 -0500 Subject: [PATCH 42/91] Attempted to factor out parts I expect to reuse for missiles and the ball. --- Machines/Atari2600/TIA.cpp | 70 ++++++++++++++++++++------------------ Machines/Atari2600/TIA.hpp | 2 ++ 2 files changed, 39 insertions(+), 33 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index b1cd3153a..b86a36b3c 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -574,47 +574,59 @@ void TIA::draw_playfield(int start, int end) } } +#pragma mark - Motion + +void TIA::perform_motion_step(int identity, int movement_time) +{ + int movement_step = (movement_time - horizontal_move_start_time_) >> 2; + if(movement_step == (motion_[identity] ^ 8)) + is_moving_[identity] = false; + else + position_[identity] ++; +} + +int TIA::perform_border_motion(int identity, int start, int end, int &movement_time) +{ + movement_time = (start + 3) & ~3; + if(!is_moving_[identity]) return 0; + + int steps_taken = 0; + int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); + // round up to the next H@1 cycle + while(is_moving_[identity] && movement_time < end && movement_time < first_pixel) + { + perform_motion_step(identity, movement_time); + movement_time += 4; + } + position_[identity] %= 160; + + return steps_taken; +} + #pragma mark - Player output void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) { int &position = position_[position_identity]; - int &motion = motion_[position_identity]; - bool &is_moving = is_moving_[position_identity]; // movement works across the entire screen, so do work that falls outside of the pixel area - int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); - int movement_time = (start + 3) & ~3; - if(is_moving) - { - // round up to the next H@1 cycle - while(movement_time < end && movement_time < first_pixel) - { - int movement = (movement_time - horizontal_move_start_time_) >> 2; - if(movement == (motion ^ 8)) - { - is_moving = false; - break; - } - position ++; - movement_time += 4; - } - position %= 160; - } + int movement_time; + perform_border_motion(position_identity, start, end, movement_time); // don't continue to do any drawing if this window ends too early + int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); if(end < first_pixel) return; if(start < first_pixel) start = first_pixel; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - if(is_moving || player.graphic) + if(is_moving_[position_identity] || player.graphic) { while(start < end) { int next_event_time = end; // is the next event a movement tick? - if(is_moving && movement_time + 4 < next_event_time) + if(is_moving_[position_identity] && movement_time + 4 < next_event_time) { next_event_time = movement_time + 4; } @@ -643,18 +655,10 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in start = next_event_time; // if the event is a motion tick, apply - if(is_moving && start == movement_time + 4) + if(is_moving_[position_identity] && start == movement_time + 4) { - int movement_step = (movement_time - horizontal_move_start_time_) >> 2; - if(movement_step == (motion ^ 8)) - { - is_moving = false; - } - else - { - position ++; - movement_time += 4; - } + perform_motion_step(position_identity, movement_time); + movement_time += 4; } // if it's a draw trigger, draw (TODO: use the actual graphic) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 28140b4b9..30ed4aff5 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -154,6 +154,8 @@ class TIA { Missile0, Missile1 }; + inline int perform_border_motion(int identity, int start, int end, int &movement_time); + inline void perform_motion_step(int identity, int movement_time); // drawing methods and state inline void output_for_cycles(int number_of_cycles); From 9ce68c38aea6e9bced25b461571d5406fda195db Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 14:01:50 -0500 Subject: [PATCH 43/91] Made an effort to implement proper pixel output for sprites. --- Machines/Atari2600/TIA.cpp | 44 +++++++++++++++++++++++++++++--------- Machines/Atari2600/TIA.hpp | 3 +++ 2 files changed, 37 insertions(+), 10 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index b86a36b3c..d351d731d 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -631,7 +631,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in next_event_time = movement_time + 4; } - // is the next event a graphics draw? + // is the next event a graphics trigger? int next_copy = 160; if(player.graphic) { @@ -650,8 +650,31 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(next_copy_time < next_event_time) next_event_time = next_copy_time; } + // maybe a deferred draw? + if(player.output_delay > 0) + { + if(start + player.output_delay < next_event_time) next_event_time = start + player.output_delay; + } + + // the decision is to progress by length + const int length = next_event_time - start; + + if(player.pixel_position < 32) + { + int adder = 4 >> player.size; +// player.pixel_position &= ~(adder - 1); + int output_cursor = 0; + while(player.pixel_position < 32 && output_cursor < length) + { + int shift = (player.pixel_position >> 2) ^ player.reverse_mask; + collision_buffer_[start + output_cursor - first_pixel_cycle] |= ((player.graphic >> shift)&1) * (uint8_t)collision_identity; + output_cursor++; + player.pixel_position += adder; + } + } + // the next interesting event is after next_event_time cycles, so progress - position = (position + next_event_time - start) % 160; + position = (position + length) % 160; start = next_event_time; // if the event is a motion tick, apply @@ -661,16 +684,17 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in movement_time += 4; } - // if it's a draw trigger, draw (TODO: use the actual graphic) + // if an output delay is being counted down, continue doing so + if(player.output_delay > 0) + { + player.output_delay -= length; + if(!player.output_delay) player.pixel_position = 0; + } + + // if it's a draw trigger, trigger a draw if(position == (next_copy % 160)) { - int cursor = start + 5 - first_pixel_cycle; - int terminus = std::min(160, cursor+8); - while(cursor < terminus) - { - collision_buffer_[cursor] |= (uint8_t)collision_identity; - cursor++; - } + player.output_delay = 5; } } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 30ed4aff5..25b214ba6 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -134,6 +134,9 @@ class TIA { int copy_flags; // a bit field, corresponding to the first few values of NUSIZ uint8_t graphic; // the player graphic int reverse_mask; // 7 for a reflected player, 0 for normal + + int pixel_position; + int output_delay; } player_[2]; // missile state From df8a5cbe6d747a17df47df8801d9f5c681aa7018 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 17:35:09 -0500 Subject: [PATCH 44/91] Made attempts (i) to respect the delay flag; and (ii) to account for border-region sprite clocking. --- Machines/Atari2600/TIA.cpp | 36 ++++++++++++++++++++++++++---------- Machines/Atari2600/TIA.hpp | 3 ++- 2 files changed, 28 insertions(+), 11 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index d351d731d..3698ddfd3 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -289,7 +289,8 @@ void TIA::set_player_number_and_size(int player, uint8_t value) void TIA::set_player_graphic(int player, uint8_t value) { - player_[player].graphic = value; + player_[player].graphic[player_[player].graphic_delay ? 1 : 0] = value; + player_[player^1].graphic[0] = player_[player^1].graphic[1]; } void TIA::set_player_reflected(int player, bool reflected) @@ -299,7 +300,7 @@ void TIA::set_player_reflected(int player, bool reflected) void TIA::set_player_delay(int player, bool delay) { - // TODO + player_[player].graphic_delay = delay; } void TIA::set_player_position(int player) @@ -611,7 +612,19 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // movement works across the entire screen, so do work that falls outside of the pixel area int movement_time; - perform_border_motion(position_identity, start, end, movement_time); + int adder = 4 >> player.size; + int added = perform_border_motion(position_identity, start, end, movement_time); + if(player.output_delay > 0) + { + int delay_distance = std::min(player.output_delay, added); + player.output_delay -= delay_distance; + added -= delay_distance; + if(!player.output_delay) player.pixel_position = 0; + } + if(player.pixel_position < 32) + { + player.pixel_position += added * adder; + } // don't continue to do any drawing if this window ends too early int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); @@ -619,7 +632,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(start < first_pixel) start = first_pixel; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - if(is_moving_[position_identity] || player.graphic) + if(is_moving_[position_identity] || player.graphic[0]) { while(start < end) { @@ -633,15 +646,19 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // is the next event a graphics trigger? int next_copy = 160; - if(player.graphic) + if(player.graphic[0]) { if(position < 16 && player.copy_flags&1) { next_copy = 16; - } else if(position < 32 && player.copy_flags&2) + } + else + if(position < 32 && player.copy_flags&2) { next_copy = 32; - } else if(position < 64 && player.copy_flags&4) + } + else + if(position < 64 && player.copy_flags&4) { next_copy = 64; } @@ -661,13 +678,12 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(player.pixel_position < 32) { - int adder = 4 >> player.size; -// player.pixel_position &= ~(adder - 1); + player.pixel_position &= ~(adder - 1); int output_cursor = 0; while(player.pixel_position < 32 && output_cursor < length) { int shift = (player.pixel_position >> 2) ^ player.reverse_mask; - collision_buffer_[start + output_cursor - first_pixel_cycle] |= ((player.graphic >> shift)&1) * (uint8_t)collision_identity; + collision_buffer_[start + output_cursor - first_pixel_cycle] |= ((player.graphic[0] >> shift)&1) * (uint8_t)collision_identity; output_cursor++; player.pixel_position += adder; } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 25b214ba6..3fa4e3a30 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -132,11 +132,12 @@ class TIA { struct Player { int size; // 0 = normal, 1 = double, 2 = quad int copy_flags; // a bit field, corresponding to the first few values of NUSIZ - uint8_t graphic; // the player graphic + uint8_t graphic[2]; // the player graphic int reverse_mask; // 7 for a reflected player, 0 for normal int pixel_position; int output_delay; + bool graphic_delay; } player_[2]; // missile state From 0c9be2b09ed6d62fb28f29b77e79e13ca6febe9d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 18:16:50 -0500 Subject: [PATCH 45/91] Shunted the collisions buffer onto a separate area of the heap for the time being, as a debugging aid. Also added a few more initial values. --- Machines/Atari2600/TIA.cpp | 13 +++++++++++-- Machines/Atari2600/TIA.hpp | 4 ++-- 2 files changed, 13 insertions(+), 4 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 3698ddfd3..984500b6b 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -25,7 +25,13 @@ TIA::TIA() : output_mode_(0), pixel_target_(nullptr), background_{0, 0}, - background_half_mask_(0) + background_half_mask_(0), + position_{0, 0, 0, 0, 0}, + motion_{0, 0, 0, 0, 0}, + is_moving_{false, false, false, false, false}, + horizontal_blank_extend_(false), + horizontal_move_start_time_(0), + collision_flags_(0) { crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); crt_->set_output_device(Outputs::CRT::Television); @@ -122,6 +128,8 @@ TIA::TIA() : colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayfieldBall; } } + + collision_buffer_.resize(160); } void TIA::set_output_mode(Atari2600::TIA::OutputMode output_mode) @@ -289,6 +297,7 @@ void TIA::set_player_number_and_size(int player, uint8_t value) void TIA::set_player_graphic(int player, uint8_t value) { + player_[player].graphic[1] = value; player_[player].graphic[player_[player].graphic_delay ? 1 : 0] = value; player_[player^1].graphic[0] = player_[player^1].graphic[1]; } @@ -393,7 +402,7 @@ void TIA::output_for_cycles(int number_of_cycles) if(!output_cursor) { - memset(collision_buffer_, 0, sizeof(collision_buffer_)); + memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_) horizontal_blank_extend_ = false; } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 3fa4e3a30..495c8431e 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -82,7 +82,8 @@ class TIA { int output_mode_; // keeps track of the target pixel buffer for this line and when it was acquired, and a corresponding collision buffer - alignas(alignof(uint32_t)) uint8_t collision_buffer_[160]; +// alignas(alignof(uint32_t)) uint8_t collision_buffer_[160]; + std::vector collision_buffer_; enum class CollisionType : uint8_t { Playfield = (1 << 0), Ball = (1 << 1), @@ -126,7 +127,6 @@ class TIA { // mirroring mode, background_[0] will be output on the left and // background_[1] on the right; otherwise background_[0] will be // output twice. - int latched_playfield_value_; // player state struct Player { From 600bdc9af7ba69230572daa052fa57d0ba28a077 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 18:18:35 -0500 Subject: [PATCH 46/91] In C++, I think the implicit cast to bool negates the need for any manual collapsing? --- Machines/Atari2600/Atari2600.cpp | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index 26b1a6c7d..64ed2e963 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -131,8 +131,8 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin } else { const uint16_t decodedAddress = address & 0x3f; switch(decodedAddress) { - case 0x00: update_video(); tia_->set_sync(!!(*value & 0x02)); break; - case 0x01: update_video(); tia_->set_blank(!!(*value & 0x02)); break; + case 0x00: update_video(); tia_->set_sync(*value & 0x02); break; + case 0x01: update_video(); tia_->set_blank(*value & 0x02); break; case 0x02: if(tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_)) @@ -161,16 +161,16 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin case 0x1b: case 0x1c: update_video(); tia_->set_player_graphic(decodedAddress - 0x1b, *value); break; case 0x1d: - case 0x1e: update_video(); tia_->set_missile_enable(decodedAddress - 0x1d, !!((*value)&2)); break; - case 0x1f: update_video(); tia_->set_ball_enable(!!((*value)&2)); break; + case 0x1e: update_video(); tia_->set_missile_enable(decodedAddress - 0x1d, (*value)&2); break; + case 0x1f: update_video(); tia_->set_ball_enable((*value)&2); break; case 0x20: case 0x21: update_video(); tia_->set_player_motion(decodedAddress - 0x20, *value); break; case 0x22: case 0x23: update_video(); tia_->set_missile_motion(decodedAddress - 0x22, *value); break; case 0x24: update_video(); tia_->set_ball_motion(*value); break; case 0x25: - case 0x26: tia_->set_player_delay(decodedAddress - 0x25, !!((*value)&1)); break; - case 0x27: tia_->set_ball_delay(!!((*value)&1)); break; + case 0x26: tia_->set_player_delay(decodedAddress - 0x25, (*value)&1); break; + case 0x27: tia_->set_ball_delay((*value)&1); break; case 0x28: case 0x29: update_video(); tia_->set_missile_position_to_player(decodedAddress - 0x28); break; case 0x2a: update_video(); tia_->move(); break; From 25776de59de2abaff059278d4984c6ff38cede47 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 19:55:02 -0500 Subject: [PATCH 47/91] I think unit testing this thing is the only way forwards. Started adding appropriate hooks. --- Machines/Atari2600/TIA.cpp | 14 ++++++++------ Machines/Atari2600/TIA.hpp | 8 ++++++++ 2 files changed, 16 insertions(+), 6 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 984500b6b..855975d8f 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -132,6 +132,11 @@ TIA::TIA() : collision_buffer_.resize(160); } +TIA::TIA(std::function line_end_function) : TIA() +{ + line_end_function_ = line_end_function; +} + void TIA::set_output_mode(Atari2600::TIA::OutputMode output_mode) { // this is the NTSC phase offset function; see below for PAL @@ -402,6 +407,7 @@ void TIA::output_for_cycles(int number_of_cycles) if(!output_cursor) { + if(line_end_function_) line_end_function_(collision_buffer_.data()); memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_) horizontal_blank_extend_ = false; } @@ -660,14 +666,10 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(position < 16 && player.copy_flags&1) { next_copy = 16; - } - else - if(position < 32 && player.copy_flags&2) + } else if(position < 32 && player.copy_flags&2) { next_copy = 32; - } - else - if(position < 64 && player.copy_flags&4) + } else if(position < 64 && player.copy_flags&4) { next_copy = 64; } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 495c8431e..dde5e06dd 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -19,6 +19,11 @@ class TIA { TIA(); ~TIA(); + // The supplied hook is for unit testing only; if instantiated with a line_end_function then it will + // be called with the latest collision buffer upon the conclusion of each line. What's a collision + // buffer? It's an implementation detail. If you're not writing a unit test, leave it alone. + TIA(std::function line_end_function); + enum class OutputMode { NTSC, PAL }; @@ -74,6 +79,7 @@ class TIA { private: std::shared_ptr crt_; + std::function line_end_function_; // the master counter; counts from 0 to 228 with all visible pixels being in the final 160 int horizontal_counter_; @@ -138,6 +144,8 @@ class TIA { int pixel_position; int output_delay; bool graphic_delay; + + Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), output_delay(0), graphic_delay(false) {} } player_[2]; // missile state From cd90118a0f3fedfa370cef54fa75f14b77ada58a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 20:32:53 -0500 Subject: [PATCH 48/91] Added two, extraordinarily simple tests. --- Machines/Atari2600/TIA.cpp | 47 ++++++----- Machines/Atari2600/TIA.hpp | 1 + .../Clock Signal.xcodeproj/project.pbxproj | 4 + OSBindings/Mac/Clock SignalTests/TIATests.mm | 78 +++++++++++++++++++ 4 files changed, 112 insertions(+), 18 deletions(-) create mode 100644 OSBindings/Mac/Clock SignalTests/TIATests.mm diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 855975d8f..a53971c30 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -19,7 +19,7 @@ namespace { uint8_t reverse_table[256]; } -TIA::TIA() : +TIA::TIA(bool create_crt) : horizontal_counter_(0), pixels_start_location_(0), output_mode_(0), @@ -33,9 +33,12 @@ TIA::TIA() : horizontal_move_start_time_(0), collision_flags_(0) { - crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); - crt_->set_output_device(Outputs::CRT::Television); - set_output_mode(OutputMode::NTSC); + if(create_crt) + { + crt_.reset(new Outputs::CRT::CRT(cycles_per_line * 2 + 1, 1, Outputs::CRT::DisplayType::NTSC60, 1)); + crt_->set_output_device(Outputs::CRT::Television); + set_output_mode(OutputMode::NTSC); + } for(int c = 0; c < 256; c++) { @@ -132,7 +135,9 @@ TIA::TIA() : collision_buffer_.resize(160); } -TIA::TIA(std::function line_end_function) : TIA() +TIA::TIA() : TIA(true) {} + +TIA::TIA(std::function line_end_function) : TIA(false) { line_end_function_ = line_end_function; } @@ -424,13 +429,13 @@ void TIA::output_for_cycles(int number_of_cycles) { \ if(horizontal_counter_ <= target) \ { \ - crt_->function((unsigned int)((horizontal_counter_ - output_cursor) * 2)); \ + if(crt_) crt_->function((unsigned int)((horizontal_counter_ - output_cursor) * 2)); \ horizontal_counter_ %= cycles_per_line; \ return; \ } \ else \ { \ - crt_->function((unsigned int)((target - output_cursor) * 2)); \ + if(crt_) crt_->function((unsigned int)((target - output_cursor) * 2)); \ output_cursor = target; \ } \ } @@ -459,16 +464,16 @@ void TIA::output_for_cycles(int number_of_cycles) if(pixel_target_) { output_pixels(pixels_start_location_, output_cursor); - crt_->output_data((unsigned int)(output_cursor - pixels_start_location_) * 2, 2); + if(crt_) crt_->output_data((unsigned int)(output_cursor - pixels_start_location_) * 2, 2); pixel_target_ = nullptr; pixels_start_location_ = 0; } int duration = std::min(228, horizontal_counter_) - output_cursor; - crt_->output_blank((unsigned int)(duration * 2)); + if(crt_) crt_->output_blank((unsigned int)(duration * 2)); } else { - if(!pixels_start_location_) + if(!pixels_start_location_ && crt_) { pixels_start_location_ = output_cursor; pixel_target_ = crt_->allocate_write_area(160); @@ -484,7 +489,7 @@ void TIA::output_for_cycles(int number_of_cycles) output_cursor++; } - if(horizontal_counter_ == cycles_per_line) + if(horizontal_counter_ == cycles_per_line && crt_) { crt_->output_data((unsigned int)(output_cursor - pixels_start_location_) * 2, 2); pixel_target_ = nullptr; @@ -549,16 +554,22 @@ void TIA::output_line() break; case sync_flag: case sync_flag | blank_flag: - crt_->output_sync(32); - crt_->output_blank(32); - crt_->output_sync(392); + if(crt_) + { + crt_->output_sync(32); + crt_->output_blank(32); + crt_->output_sync(392); + } horizontal_blank_extend_ = false; break; case blank_flag: - crt_->output_blank(32); - crt_->output_sync(32); - crt_->output_default_colour_burst(32); - crt_->output_blank(360); + if(crt_) + { + crt_->output_blank(32); + crt_->output_sync(32); + crt_->output_default_colour_burst(32); + crt_->output_blank(360); + } horizontal_blank_extend_ = false; break; } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index dde5e06dd..c54eb50d4 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -23,6 +23,7 @@ class TIA { // be called with the latest collision buffer upon the conclusion of each line. What's a collision // buffer? It's an implementation detail. If you're not writing a unit test, leave it alone. TIA(std::function line_end_function); + TIA(bool create_crt); enum class OutputMode { NTSC, PAL diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 08c8373f8..e4c35ff0d 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -30,6 +30,7 @@ 4B2A53A11D117D36003C6002 /* CSAtari2600.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B2A539A1D117D36003C6002 /* CSAtari2600.mm */; }; 4B2A53A21D117D36003C6002 /* CSElectron.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B2A539C1D117D36003C6002 /* CSElectron.mm */; }; 4B2A53A31D117D36003C6002 /* CSVic20.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B2A539E1D117D36003C6002 /* CSVic20.mm */; }; + 4B2AF8691E513FC20027EE29 /* TIATests.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B2AF8681E513FC20027EE29 /* TIATests.mm */; }; 4B2BFC5F1D613E0200BA3AA9 /* TapePRG.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B2BFC5D1D613E0200BA3AA9 /* TapePRG.cpp */; }; 4B2BFDB21DAEF5FF001A68B8 /* Video.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B2BFDB01DAEF5FF001A68B8 /* Video.cpp */; }; 4B2C45421E3C3896002A2389 /* cartridge.png in Resources */ = {isa = PBXBuildFile; fileRef = 4B2C45411E3C3896002A2389 /* cartridge.png */; }; @@ -474,6 +475,7 @@ 4B2A539C1D117D36003C6002 /* CSElectron.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = CSElectron.mm; sourceTree = ""; }; 4B2A539D1D117D36003C6002 /* CSVic20.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = CSVic20.h; sourceTree = ""; }; 4B2A539E1D117D36003C6002 /* CSVic20.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = CSVic20.mm; sourceTree = ""; }; + 4B2AF8681E513FC20027EE29 /* TIATests.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = TIATests.mm; sourceTree = ""; }; 4B2BFC5D1D613E0200BA3AA9 /* TapePRG.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = TapePRG.cpp; sourceTree = ""; }; 4B2BFC5E1D613E0200BA3AA9 /* TapePRG.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = TapePRG.hpp; sourceTree = ""; }; 4B2BFDB01DAEF5FF001A68B8 /* Video.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = Video.cpp; path = Oric/Video.cpp; sourceTree = ""; }; @@ -1724,6 +1726,7 @@ 4B121F941E05E66800BFDA12 /* PCMPatchedTrackTests.mm */, 4B121F9A1E06293F00BFDA12 /* PCMSegmentEventSourceTests.mm */, 4BD4A8CF1E077FD20020D856 /* PCMTrackTests.mm */, + 4B2AF8681E513FC20027EE29 /* TIATests.mm */, 4B1D08051E0F7A1100763741 /* TimeTests.mm */, 4BB73EB81B587A5100552FC2 /* Info.plist */, 4BC9E1ED1D23449A003FCEE4 /* 6502InterruptTests.swift */, @@ -2494,6 +2497,7 @@ 4BC9E1EE1D23449A003FCEE4 /* 6502InterruptTests.swift in Sources */, 4BEF6AAA1D35CE9E00E73575 /* DigitalPhaseLockedLoopBridge.mm in Sources */, 4B50730A1DDFCFDF00C48FBD /* ArrayBuilderTests.mm in Sources */, + 4B2AF8691E513FC20027EE29 /* TIATests.mm in Sources */, 4B3BA0CE1D318B44005DD7A7 /* C1540Bridge.mm in Sources */, 4B3BA0D11D318B44005DD7A7 /* TestMachine.mm in Sources */, 4B92EACA1B7C112B00246143 /* 6502TimingTests.swift in Sources */, diff --git a/OSBindings/Mac/Clock SignalTests/TIATests.mm b/OSBindings/Mac/Clock SignalTests/TIATests.mm new file mode 100644 index 000000000..412f54a55 --- /dev/null +++ b/OSBindings/Mac/Clock SignalTests/TIATests.mm @@ -0,0 +1,78 @@ +// +// TIATests.m +// Clock Signal +// +// Created by Thomas Harte on 12/02/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#import + +#include "TIA.hpp" + +static uint8_t *line; +static void receive_line(uint8_t *next_line) +{ + line = next_line; +} + +@interface TIATests : XCTestCase +@end + +@implementation TIATests { + std::unique_ptr _tia; +} + +- (void)setUp +{ + [super setUp]; + std::function function = receive_line; + _tia.reset(new Atari2600::TIA(function)); + line = nullptr; +} + +- (void)testReflectedPlayfield +{ + // set reflected, bit pattern 1000 + _tia->set_playfield_control_and_ball_size(1); + _tia->set_playfield(0, 0x10); + _tia->set_playfield(1, 0xf0); + _tia->set_playfield(2, 0x0e); + _tia->run_for_cycles(228); + + XCTAssert(line != nullptr, @"228 cycles should have ended the line"); + + uint8_t expected_line[] = { + 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1 + }; + XCTAssert(!memcmp(expected_line, line, sizeof(expected_line))); +} + +- (void)testRepeatedPlayfield +{ + // set reflected, bit pattern 1000 + _tia->set_playfield_control_and_ball_size(0); + _tia->set_playfield(0, 0x10); + _tia->set_playfield(1, 0xf0); + _tia->set_playfield(2, 0x0e); + _tia->run_for_cycles(228); + + XCTAssert(line != nullptr, @"228 cycles should have ended the line"); + + uint8_t expected_line[] = { + 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + }; + XCTAssert(!memcmp(expected_line, line, sizeof(expected_line))); +} + +@end From dd174596874f4aa648528f1172cf675931a58145 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 12 Feb 2017 20:42:49 -0500 Subject: [PATCH 49/91] Added my first failing test: delay is incorrect when resetting outside of the play area. --- OSBindings/Mac/Clock SignalTests/TIATests.mm | 43 +++++++++++++++++++- 1 file changed, 42 insertions(+), 1 deletion(-) diff --git a/OSBindings/Mac/Clock SignalTests/TIATests.mm b/OSBindings/Mac/Clock SignalTests/TIATests.mm index 412f54a55..5ebaee5fe 100644 --- a/OSBindings/Mac/Clock SignalTests/TIATests.mm +++ b/OSBindings/Mac/Clock SignalTests/TIATests.mm @@ -29,6 +29,15 @@ static void receive_line(uint8_t *next_line) std::function function = receive_line; _tia.reset(new Atari2600::TIA(function)); line = nullptr; + + _tia->set_playfield(0, 0x00); + _tia->set_playfield(1, 0x00); + _tia->set_playfield(2, 0x00); + _tia->set_player_graphic(0, 0x00); + _tia->set_player_graphic(1, 0x00); + _tia->set_ball_enable(false); + _tia->set_missile_enable(0, false); + _tia->set_missile_enable(1, false); } - (void)testReflectedPlayfield @@ -60,8 +69,8 @@ static void receive_line(uint8_t *next_line) _tia->set_playfield(0, 0x10); _tia->set_playfield(1, 0xf0); _tia->set_playfield(2, 0x0e); - _tia->run_for_cycles(228); + _tia->run_for_cycles(228); XCTAssert(line != nullptr, @"228 cycles should have ended the line"); uint8_t expected_line[] = { @@ -75,4 +84,36 @@ static void receive_line(uint8_t *next_line) XCTAssert(!memcmp(expected_line, line, sizeof(expected_line))); } +- (void)testSinglePlayer +{ + // set a player graphic, reset position so that it'll appear from column 1 + _tia->set_player_graphic(0, 0xff); + _tia->set_player_position(0); + + _tia->run_for_cycles(228); + uint8_t first_expected_line[] = { + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + }; + XCTAssert(line != nullptr, @"228 cycles should have ended the line"); + XCTAssert(!memcmp(first_expected_line, line, sizeof(first_expected_line))); + line = nullptr; + + _tia->run_for_cycles(228); + uint8_t second_expected_line[] = { + 0, 4, 4, 4, 4, 4, 4, 4, 4, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + }; + XCTAssert(line != nullptr, @"228 cycles should have ended the line"); + XCTAssert(!memcmp(second_expected_line, line, sizeof(second_expected_line))); +} + @end From b5357860b9ddd83ee5ce08fb4db25e70a34c7c06 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 14 Feb 2017 20:56:16 -0500 Subject: [PATCH 50/91] Made an attempt to split things apart so as to be able to introduce the proper sprite latency. --- Machines/Atari2600/TIA.cpp | 83 +++++++++++++++++--------------------- Machines/Atari2600/TIA.hpp | 4 +- 2 files changed, 40 insertions(+), 47 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index a53971c30..8830208d8 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -418,7 +418,9 @@ void TIA::output_for_cycles(int number_of_cycles) } // accumulate an OR'd version of the output into the collision buffer - draw_playfield(output_cursor, horizontal_counter_); + int latent_start = output_cursor + 4; + int latent_end = horizontal_counter_ + 4; + draw_playfield(latent_start, latent_end); draw_player(player_[0], CollisionType::Player0, (int)MotionIndex::Player0, output_cursor, horizontal_counter_); draw_player(player_[1], CollisionType::Player1, (int)MotionIndex::Player1, output_cursor, horizontal_counter_); @@ -580,11 +582,7 @@ void TIA::output_line() void TIA::draw_playfield(int start, int end) { // don't do anything if this window ends too early - if(end < first_pixel_cycle - 4) return; - - // look at what needs to be output four cycles into the future, to model playfield output latency - start += 4; - end += 4; + if(end < first_pixel_cycle) return; // clip to drawable bounds start = std::max(start, first_pixel_cycle); @@ -618,7 +616,7 @@ int TIA::perform_border_motion(int identity, int start, int end, int &movement_t if(!is_moving_[identity]) return 0; int steps_taken = 0; - int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); + int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0) - 4; // round up to the next H@1 cycle while(is_moving_[identity] && movement_time < end && movement_time < first_pixel) { @@ -632,30 +630,10 @@ int TIA::perform_border_motion(int identity, int start, int end, int &movement_t #pragma mark - Player output -void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) +void TIA::draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end, int &movement_time) { int &position = position_[position_identity]; - - // movement works across the entire screen, so do work that falls outside of the pixel area - int movement_time; int adder = 4 >> player.size; - int added = perform_border_motion(position_identity, start, end, movement_time); - if(player.output_delay > 0) - { - int delay_distance = std::min(player.output_delay, added); - player.output_delay -= delay_distance; - added -= delay_distance; - if(!player.output_delay) player.pixel_position = 0; - } - if(player.pixel_position < 32) - { - player.pixel_position += added * adder; - } - - // don't continue to do any drawing if this window ends too early - int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0); - if(end < first_pixel) return; - if(start < first_pixel) start = first_pixel; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion if(is_moving_[position_identity] || player.graphic[0]) @@ -689,23 +667,17 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(next_copy_time < next_event_time) next_event_time = next_copy_time; } - // maybe a deferred draw? - if(player.output_delay > 0) - { - if(start + player.output_delay < next_event_time) next_event_time = start + player.output_delay; - } - // the decision is to progress by length const int length = next_event_time - start; - if(player.pixel_position < 32) + if(player.pixel_position < 36) { player.pixel_position &= ~(adder - 1); int output_cursor = 0; - while(player.pixel_position < 32 && output_cursor < length) + while(player.pixel_position < 36 && output_cursor < length) { int shift = (player.pixel_position >> 2) ^ player.reverse_mask; - collision_buffer_[start + output_cursor - first_pixel_cycle] |= ((player.graphic[0] >> shift)&1) * (uint8_t)collision_identity; + collision_buffer_[start + output_cursor] |= ((player.graphic[0] >> shift)&1) * (uint8_t)collision_identity; output_cursor++; player.pixel_position += adder; } @@ -722,17 +694,10 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in movement_time += 4; } - // if an output delay is being counted down, continue doing so - if(player.output_delay > 0) - { - player.output_delay -= length; - if(!player.output_delay) player.pixel_position = 0; - } - // if it's a draw trigger, trigger a draw if(position == (next_copy % 160)) { - player.output_delay = 5; + player.pixel_position = 0; } } } @@ -742,3 +707,31 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in position = (position + end - start) % 160; } } + +void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) +{ + int movement_time; + int adder = 4 >> player.size; + + // movement works across the entire screen, so do work that falls outside of the pixel area + player.pixel_position += adder * perform_border_motion(position_identity, start, end, movement_time);; + + // don't continue to do any drawing if this window ends too early + int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); + if(end < first_pixel) return; + if(start < first_pixel) start = first_pixel; + + // perform the visible part of the line, if any + if(start < 224) + { + movement_time -= first_pixel_cycle - 4; + draw_player_visible(player, collision_identity, position_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160), movement_time); + } + + // move further if required + if(is_moving_[position_identity] && end >= 224 && movement_time < end) + { + perform_motion_step(position_identity, movement_time); + player.pixel_position += adder; + } +} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index c54eb50d4..0903f56eb 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -143,10 +143,9 @@ class TIA { int reverse_mask; // 7 for a reflected player, 0 for normal int pixel_position; - int output_delay; bool graphic_delay; - Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), output_delay(0), graphic_delay(false) {} + Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_delay(false) {} } player_[2]; // missile state @@ -176,6 +175,7 @@ class TIA { inline void draw_playfield(int start, int end); inline void draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); + inline void draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end, int &movement_time); inline void update_motion(int start, int end); From 09309aa74fdd31acab2dec519f3916aab67bbe7e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 16 Feb 2017 18:52:39 -0500 Subject: [PATCH 51/91] Attempted to prevent extraneous moves. --- Machines/Atari2600/TIA.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 8830208d8..a64915bf0 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -726,6 +726,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in { movement_time -= first_pixel_cycle - 4; draw_player_visible(player, collision_identity, position_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160), movement_time); + movement_time += first_pixel_cycle - 4; } // move further if required From 5ea232310ffdd53008f2d0ad0b69962c8bf0f1ad Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 16 Feb 2017 18:55:58 -0500 Subject: [PATCH 52/91] Added a check against negative runs. --- Machines/Atari2600/TIA.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index a64915bf0..2915640f3 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -720,6 +720,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); if(end < first_pixel) return; if(start < first_pixel) start = first_pixel; + if(start >= end) return; // perform the visible part of the line, if any if(start < 224) From 8d502a0b0354f141e5123d40535c4636edf97ba3 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 16 Feb 2017 20:28:37 -0500 Subject: [PATCH 53/91] Decided not to run before I can walk and switched to storing the motion time and next step explicitly per object. --- Machines/Atari2600/TIA.cpp | 58 ++++++++++++++++++++------------------ Machines/Atari2600/TIA.hpp | 11 ++++---- 2 files changed, 36 insertions(+), 33 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 2915640f3..c59cffeb2 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -30,7 +30,6 @@ TIA::TIA(bool create_crt) : motion_{0, 0, 0, 0, 0}, is_moving_{false, false, false, false, false}, horizontal_blank_extend_(false), - horizontal_move_start_time_(0), collision_flags_(0) { if(create_crt) @@ -329,7 +328,7 @@ void TIA::set_player_position(int player) void TIA::set_player_motion(int player, uint8_t motion) { - motion_[(int)MotionIndex::Player0 + player] = motion >> 4; + motion_[(int)MotionIndex::Player0 + player] = (motion >> 4)&0xf; } void TIA::set_player_missile_colour(int player, uint8_t colour) @@ -373,7 +372,8 @@ void TIA::move() { horizontal_blank_extend_ = true; is_moving_[0] = is_moving_[1] = is_moving_[2] = is_moving_[3] = is_moving_[4] = true; - horizontal_move_start_time_ = horizontal_counter_; + motion_step_[0] = motion_step_[1] = motion_step_[2] = motion_step_[3] = motion_step_[4] = 15; + motion_time_[0] = motion_time_[1] = motion_time_[2] = motion_time_[3] = motion_time_[4] = (horizontal_counter_ + 3) & ~3; } void TIA::clear_motion() @@ -415,6 +415,7 @@ void TIA::output_for_cycles(int number_of_cycles) if(line_end_function_) line_end_function_(collision_buffer_.data()); memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_) horizontal_blank_extend_ = false; + for(int c = 0; c < 5; c++) motion_time_[c] %= 228; } // accumulate an OR'd version of the output into the collision buffer @@ -601,27 +602,27 @@ void TIA::draw_playfield(int start, int end) #pragma mark - Motion -void TIA::perform_motion_step(int identity, int movement_time) +void TIA::perform_motion_step(int identity) { - int movement_step = (movement_time - horizontal_move_start_time_) >> 2; - if(movement_step == (motion_[identity] ^ 8)) + if((motion_step_[identity] ^ (motion_[identity] ^ 8)) == 0xf) is_moving_[identity] = false; else + { position_[identity] ++; + motion_step_[identity] --; + motion_time_[identity] += 4; + } } -int TIA::perform_border_motion(int identity, int start, int end, int &movement_time) +int TIA::perform_border_motion(int identity, int start, int end) { - movement_time = (start + 3) & ~3; if(!is_moving_[identity]) return 0; int steps_taken = 0; - int first_pixel = first_pixel_cycle + (horizontal_blank_extend_ ? 8 : 0) - 4; - // round up to the next H@1 cycle - while(is_moving_[identity] && movement_time < end && movement_time < first_pixel) + while(is_moving_[identity] && motion_time_[identity] < end) { - perform_motion_step(identity, movement_time); - movement_time += 4; + perform_motion_step(identity); + steps_taken++; } position_[identity] %= 160; @@ -630,7 +631,7 @@ int TIA::perform_border_motion(int identity, int start, int end, int &movement_t #pragma mark - Player output -void TIA::draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end, int &movement_time) +void TIA::draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) { int &position = position_[position_identity]; int adder = 4 >> player.size; @@ -638,14 +639,15 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion if(is_moving_[position_identity] || player.graphic[0]) { + int next_motion_time = motion_time_[position_identity] - first_pixel_cycle + 4; while(start < end) { int next_event_time = end; // is the next event a movement tick? - if(is_moving_[position_identity] && movement_time + 4 < next_event_time) + if(is_moving_[position_identity] && next_motion_time < next_event_time) { - next_event_time = movement_time + 4; + next_event_time = next_motion_time; } // is the next event a graphics trigger? @@ -688,10 +690,10 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, start = next_event_time; // if the event is a motion tick, apply - if(is_moving_[position_identity] && start == movement_time + 4) + if(is_moving_[position_identity] && start == next_motion_time) { - perform_motion_step(position_identity, movement_time); - movement_time += 4; + perform_motion_step(position_identity); + next_motion_time += 4; } // if it's a draw trigger, trigger a draw @@ -710,14 +712,16 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) { - int movement_time; int adder = 4 >> player.size; + int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); // movement works across the entire screen, so do work that falls outside of the pixel area - player.pixel_position += adder * perform_border_motion(position_identity, start, end, movement_time);; + if(start < first_pixel) + { + player.pixel_position = std::min(36, player.pixel_position + adder * perform_border_motion(position_identity, start, std::max(end, first_pixel))); + } // don't continue to do any drawing if this window ends too early - int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); if(end < first_pixel) return; if(start < first_pixel) start = first_pixel; if(start >= end) return; @@ -725,15 +729,13 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // perform the visible part of the line, if any if(start < 224) { - movement_time -= first_pixel_cycle - 4; - draw_player_visible(player, collision_identity, position_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160), movement_time); - movement_time += first_pixel_cycle - 4; + draw_player_visible(player, collision_identity, position_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); } // move further if required - if(is_moving_[position_identity] && end >= 224 && movement_time < end) + if(is_moving_[position_identity] && end >= 224 && motion_time_[position_identity] < end) { - perform_motion_step(position_identity, movement_time); - player.pixel_position += adder; + perform_motion_step(position_identity); + player.pixel_position = std::min(36, player.pixel_position + adder); } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 0903f56eb..78cf46f5b 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -145,7 +145,7 @@ class TIA { int pixel_position; bool graphic_delay; - Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_delay(false) {} + Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(36), graphic_delay(false) {} } player_[2]; // missile state @@ -155,8 +155,9 @@ class TIA { // movement bool horizontal_blank_extend_; - int horizontal_move_start_time_; int motion_[5]; + int motion_step_[5]; + int motion_time_[5]; int position_[5]; bool is_moving_[5]; enum class MotionIndex : uint8_t { @@ -166,8 +167,8 @@ class TIA { Missile0, Missile1 }; - inline int perform_border_motion(int identity, int start, int end, int &movement_time); - inline void perform_motion_step(int identity, int movement_time); + inline int perform_border_motion(int identity, int start, int end); + inline void perform_motion_step(int identity); // drawing methods and state inline void output_for_cycles(int number_of_cycles); @@ -175,7 +176,7 @@ class TIA { inline void draw_playfield(int start, int end); inline void draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); - inline void draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end, int &movement_time); + inline void draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); inline void update_motion(int start, int end); From ed5ff49ef5575eeccfbdfb511bf20ee8db9579f8 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 16 Feb 2017 20:52:01 -0500 Subject: [PATCH 54/91] Fixed vertical delay, retreated from my previous thought about adding the one extra cycle of sprite delay, at least temporarily. --- Machines/Atari2600/TIA.cpp | 17 ++++++++--------- Machines/Atari2600/TIA.hpp | 6 +++--- 2 files changed, 11 insertions(+), 12 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index c59cffeb2..6cbaa663a 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -307,7 +307,6 @@ void TIA::set_player_number_and_size(int player, uint8_t value) void TIA::set_player_graphic(int player, uint8_t value) { player_[player].graphic[1] = value; - player_[player].graphic[player_[player].graphic_delay ? 1 : 0] = value; player_[player^1].graphic[0] = player_[player^1].graphic[1]; } @@ -318,7 +317,7 @@ void TIA::set_player_reflected(int player, bool reflected) void TIA::set_player_delay(int player, bool delay) { - player_[player].graphic_delay = delay; + player_[player].graphic_index = delay ? 0 : 1; } void TIA::set_player_position(int player) @@ -637,7 +636,7 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, int adder = 4 >> player.size; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - if(is_moving_[position_identity] || player.graphic[0]) + if(is_moving_[position_identity] || player.graphic[player.graphic_index]) { int next_motion_time = motion_time_[position_identity] - first_pixel_cycle + 4; while(start < end) @@ -652,7 +651,7 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, // is the next event a graphics trigger? int next_copy = 160; - if(player.graphic[0]) + if(player.graphic[player.graphic_index]) { if(position < 16 && player.copy_flags&1) { @@ -672,14 +671,14 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, // the decision is to progress by length const int length = next_event_time - start; - if(player.pixel_position < 36) + if(player.pixel_position < 32) { player.pixel_position &= ~(adder - 1); int output_cursor = 0; - while(player.pixel_position < 36 && output_cursor < length) + while(player.pixel_position < 32 && output_cursor < length) { int shift = (player.pixel_position >> 2) ^ player.reverse_mask; - collision_buffer_[start + output_cursor] |= ((player.graphic[0] >> shift)&1) * (uint8_t)collision_identity; + collision_buffer_[start + output_cursor] |= ((player.graphic[player.graphic_index] >> shift)&1) * (uint8_t)collision_identity; output_cursor++; player.pixel_position += adder; } @@ -718,7 +717,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // movement works across the entire screen, so do work that falls outside of the pixel area if(start < first_pixel) { - player.pixel_position = std::min(36, player.pixel_position + adder * perform_border_motion(position_identity, start, std::max(end, first_pixel))); + player.pixel_position = std::min(32, player.pixel_position + adder * perform_border_motion(position_identity, start, std::max(end, first_pixel))); } // don't continue to do any drawing if this window ends too early @@ -736,6 +735,6 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in if(is_moving_[position_identity] && end >= 224 && motion_time_[position_identity] < end) { perform_motion_step(position_identity); - player.pixel_position = std::min(36, player.pixel_position + adder); + player.pixel_position = std::min(32, player.pixel_position + adder); } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 78cf46f5b..903afdc53 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -139,13 +139,13 @@ class TIA { struct Player { int size; // 0 = normal, 1 = double, 2 = quad int copy_flags; // a bit field, corresponding to the first few values of NUSIZ - uint8_t graphic[2]; // the player graphic + uint8_t graphic[2]; // the player graphic; 1 = new, 0 = current int reverse_mask; // 7 for a reflected player, 0 for normal + int graphic_index; int pixel_position; - bool graphic_delay; - Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(36), graphic_delay(false) {} + Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0) {} } player_[2]; // missile state From cd7876a746ec39ca29b5f623c3b857114aeced46 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 18 Feb 2017 20:18:50 -0500 Subject: [PATCH 55/91] Reintroduced the extra clocking delay. --- Machines/Atari2600/TIA.cpp | 14 +++++++++++++- Machines/Atari2600/TIA.hpp | 3 ++- 2 files changed, 15 insertions(+), 2 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 6cbaa663a..b653e8833 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -649,6 +649,11 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, next_event_time = next_motion_time; } + if(player.pixel_delay && start + player.pixel_delay < next_event_time) + { + next_event_time = start + player.pixel_delay; + } + // is the next event a graphics trigger? int next_copy = 160; if(player.graphic[player.graphic_index]) @@ -695,10 +700,17 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, next_motion_time += 4; } + // possibly it's a pixel delay + if(player.pixel_delay) + { + player.pixel_delay -= length; + if(!player.pixel_delay) player.pixel_position = 0; + } + // if it's a draw trigger, trigger a draw if(position == (next_copy % 160)) { - player.pixel_position = 0; + player.pixel_delay = 1; } } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 903afdc53..02a2e3ab9 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -144,8 +144,9 @@ class TIA { int graphic_index; int pixel_position; + int pixel_delay; - Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0) {} + Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0), pixel_delay(0) {} } player_[2]; // missile state From 04693b067c4fb52623e5fce1a1712f452d25ea9b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 18 Feb 2017 21:36:48 -0500 Subject: [PATCH 56/91] Fixed failure of the optimised route to pump the pixel clock; removed optimisation entirely for now. --- Machines/Atari2600/TIA.cpp | 18 ++++++++++-------- 1 file changed, 10 insertions(+), 8 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index b653e8833..ec2df4b24 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -636,8 +636,8 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, int adder = 4 >> player.size; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - if(is_moving_[position_identity] || player.graphic[player.graphic_index]) - { +// if(is_moving_[position_identity] || player.graphic[player.graphic_index]) +// { int next_motion_time = motion_time_[position_identity] - first_pixel_cycle + 4; while(start < end) { @@ -713,12 +713,14 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, player.pixel_delay = 1; } } - } - else - { - // just advance the timer all in one jump - position = (position + end - start) % 160; - } +// } +// else +// { +// // just advance the timer all in one jump +// const int length = end - start; +// position = (position + length) % 160; +// player.pixel_position = std::min(32, player.pixel_position + adder * length); +// } } void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) From 53cd1257125eb664135457d97845a93690a13a87 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 19 Feb 2017 07:28:24 -0500 Subject: [PATCH 57/91] Added stub calls to draw the missiles and ball. --- Machines/Atari2600/TIA.cpp | 44 ++++++++++++++++++++++++++------------ Machines/Atari2600/TIA.hpp | 10 +++++++-- 2 files changed, 38 insertions(+), 16 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index ec2df4b24..5e6acf1a2 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -322,7 +322,12 @@ void TIA::set_player_delay(int player, bool delay) void TIA::set_player_position(int player) { - position_[(int)MotionIndex::Player0 + player] = 0; + // players have an extra clock of delay before output and don't display upon reset; + // both aims are achieved by setting to -1 because: (i) it causes the clock to be + // one behind its real hardware value, creating the extra delay; and (ii) the player + // code is written to start a draw upon wraparound from 159 to 0, so -1 is the + // correct option rather than 159. + position_[(int)MotionIndex::Player0 + player] = -1; } void TIA::set_player_motion(int player, uint8_t motion) @@ -423,6 +428,9 @@ void TIA::output_for_cycles(int number_of_cycles) draw_playfield(latent_start, latent_end); draw_player(player_[0], CollisionType::Player0, (int)MotionIndex::Player0, output_cursor, horizontal_counter_); draw_player(player_[1], CollisionType::Player1, (int)MotionIndex::Player1, output_cursor, horizontal_counter_); + draw_missile(missile_[0], CollisionType::Missile0, (int)MotionIndex::Missile0, output_cursor, horizontal_counter_); + draw_missile(missile_[1], CollisionType::Missile1, (int)MotionIndex::Missile1, output_cursor, horizontal_counter_); + draw_ball(output_cursor, horizontal_counter_); // convert to television signals @@ -649,11 +657,6 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, next_event_time = next_motion_time; } - if(player.pixel_delay && start + player.pixel_delay < next_event_time) - { - next_event_time = start + player.pixel_delay; - } - // is the next event a graphics trigger? int next_copy = 160; if(player.graphic[player.graphic_index]) @@ -700,17 +703,10 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, next_motion_time += 4; } - // possibly it's a pixel delay - if(player.pixel_delay) - { - player.pixel_delay -= length; - if(!player.pixel_delay) player.pixel_position = 0; - } - // if it's a draw trigger, trigger a draw if(position == (next_copy % 160)) { - player.pixel_delay = 1; + player.pixel_position = 0; } } // } @@ -752,3 +748,23 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in player.pixel_position = std::min(32, player.pixel_position + adder); } } + +#pragma mark - Missile output + +void TIA::draw_missile(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end) +{ +} + +void TIA::draw_missile_visible(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end) +{ +} + +#pragma mark - Ball output + +void TIA::draw_ball(int start, int end) +{ +} + +void TIA::draw_ball_visible(int start, int end) +{ +} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 02a2e3ab9..77e52729d 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -144,9 +144,8 @@ class TIA { int graphic_index; int pixel_position; - int pixel_delay; - Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0), pixel_delay(0) {} + Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0) {} } player_[2]; // missile state @@ -176,9 +175,16 @@ class TIA { inline void output_line(); inline void draw_playfield(int start, int end); + inline void draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); inline void draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); + inline void draw_missile(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end); + inline void draw_missile_visible(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end); + + inline void draw_ball(int start, int end); + inline void draw_ball_visible(int start, int end); + inline void update_motion(int start, int end); int pixels_start_location_; From e00339ef0a28729a0446382e7babd54f28063ff2 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 19 Feb 2017 08:02:54 -0500 Subject: [PATCH 58/91] Attempted to reintroduce the ball. --- Machines/Atari2600/TIA.cpp | 95 ++++++++++++++++++++++++++++++++++++++ Machines/Atari2600/TIA.hpp | 11 +++++ 2 files changed, 106 insertions(+) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 5e6acf1a2..0939ab090 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -272,6 +272,8 @@ void TIA::set_playfield_control_and_ball_size(uint8_t value) playfield_priority_ = PlayfieldPriority::OnTop; break; } + + ball_.size = 1 << ((value >> 4)&3); } void TIA::set_playfield_ball_colour(uint8_t colour) @@ -308,6 +310,7 @@ void TIA::set_player_graphic(int player, uint8_t value) { player_[player].graphic[1] = value; player_[player^1].graphic[0] = player_[player^1].graphic[1]; + if(player) ball_.enabled[0] = ball_.enabled[1]; } void TIA::set_player_reflected(int player, bool reflected) @@ -358,18 +361,25 @@ void TIA::set_missile_motion(int missile, uint8_t motion) void TIA::set_ball_enable(bool enabled) { + ball_.enabled[1] = enabled; } void TIA::set_ball_delay(bool delay) { + ball_.enabled_index = delay ? 0 : 1; } void TIA::set_ball_position() { + position_[(int)MotionIndex::Ball] = 0; + + // setting the ball position also triggers a draw + ball_.pixel_position = ball_.size; } void TIA::set_ball_motion(uint8_t motion) { + motion_[(int)MotionIndex::Ball] = (motion >> 4) & 0xf; } void TIA::move() @@ -763,8 +773,93 @@ void TIA::draw_missile_visible(Missile &missile, CollisionType collision_identit void TIA::draw_ball(int start, int end) { + int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); + + // movement works across the entire screen, so do work that falls outside of the pixel area + if(start < first_pixel) + { + ball_.pixel_position = std::max(0, ball_.pixel_position - perform_border_motion((int)MotionIndex::Ball, start, std::max(end, first_pixel))); + } + + // don't continue to do any drawing if this window ends too early + if(end < first_pixel) return; + if(start < first_pixel) start = first_pixel; + if(start >= end) return; + + // perform the visible part of the line, if any + if(start < 224) + { + draw_ball_visible(start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); + } + + // move further if required + if(is_moving_[(int)MotionIndex::Ball] && end >= 224 && motion_time_[(int)MotionIndex::Ball] < end) + { + perform_motion_step((int)MotionIndex::Ball); + ball_.pixel_position = std::max(0, ball_.pixel_position - 1); + } } void TIA::draw_ball_visible(int start, int end) { + int &position = position_[(int)MotionIndex::Ball]; + + // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion + int next_motion_time = motion_time_[(int)MotionIndex::Ball] - first_pixel_cycle + 4; + while(start < end) + { + int next_event_time = end; + + // is the next event a movement tick? + if(is_moving_[(int)MotionIndex::Ball] && next_motion_time < next_event_time) + { + next_event_time = next_motion_time; + } + + // is the next event a graphics trigger? + if(ball_.enabled[ball_.enabled_index]) + { + int time_until_copy = 160 - position; + int next_copy_time = start + time_until_copy; + if(next_copy_time < next_event_time) next_event_time = next_copy_time; + } + + // the decision is to progress by length + const int length = next_event_time - start; + + if(ball_.pixel_position) + { + int output_cursor = 0; + if(ball_.enabled[ball_.enabled_index]) + { + while(ball_.pixel_position && output_cursor < length) + { + collision_buffer_[start + output_cursor] |= (uint8_t)CollisionType::Ball; + output_cursor++; + ball_.pixel_position--; + } + } + else + { + ball_.pixel_position = std::max(0, ball_.pixel_position - length); + } + } + + // the next interesting event is after next_event_time cycles, so progress + position = (position + length) % 160; + start = next_event_time; + + // if the event is a motion tick, apply + if(is_moving_[(int)MotionIndex::Ball] && start == next_motion_time) + { + perform_motion_step((int)MotionIndex::Ball); + next_motion_time += 4; + } + + // if it's a draw trigger, trigger a draw + if(!position) + { + ball_.pixel_position = ball_.size; + } + } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 77e52729d..2bb5af4bc 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -153,6 +153,17 @@ class TIA { int size; // 0 = 1 pixel, 1 = 2 pixels, etc } missile_[2]; + // ball state + struct Ball { + bool enabled[2]; + int enabled_index; + int size; + + int pixel_position; + + Ball() : pixel_position(0), size(1), enabled_index(0) {} + } ball_; + // movement bool horizontal_blank_extend_; int motion_[5]; From 51bcaea60c414f5e4686a1a687352fc90d69459b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 19 Feb 2017 12:00:04 -0500 Subject: [PATCH 59/91] Disabled incorrect 'optimisations'. --- Machines/Atari2600/TIA.cpp | 106 +++++++++++++++++-------------------- 1 file changed, 50 insertions(+), 56 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 0939ab090..fd5ebb3fd 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -654,44 +654,41 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, int adder = 4 >> player.size; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion -// if(is_moving_[position_identity] || player.graphic[player.graphic_index]) -// { - int next_motion_time = motion_time_[position_identity] - first_pixel_cycle + 4; - while(start < end) + int next_motion_time = motion_time_[position_identity] - first_pixel_cycle + 4; + while(start < end) + { + int next_event_time = end; + + // is the next event a movement tick? + if(is_moving_[position_identity] && next_motion_time < next_event_time) { - int next_event_time = end; + next_event_time = next_motion_time; + } - // is the next event a movement tick? - if(is_moving_[position_identity] && next_motion_time < next_event_time) - { - next_event_time = next_motion_time; - } + // is the next event a graphics trigger? + int next_copy = 160; + if(position < 16 && player.copy_flags&1) + { + next_copy = 16; + } else if(position < 32 && player.copy_flags&2) + { + next_copy = 32; + } else if(position < 64 && player.copy_flags&4) + { + next_copy = 64; + } - // is the next event a graphics trigger? - int next_copy = 160; + int next_copy_time = start + next_copy - position; + if(next_copy_time < next_event_time) next_event_time = next_copy_time; + + // the decision is to progress by length + const int length = next_event_time - start; + + if(player.pixel_position < 32) + { + player.pixel_position &= ~(adder - 1); if(player.graphic[player.graphic_index]) { - if(position < 16 && player.copy_flags&1) - { - next_copy = 16; - } else if(position < 32 && player.copy_flags&2) - { - next_copy = 32; - } else if(position < 64 && player.copy_flags&4) - { - next_copy = 64; - } - - int next_copy_time = start + next_copy - position; - if(next_copy_time < next_event_time) next_event_time = next_copy_time; - } - - // the decision is to progress by length - const int length = next_event_time - start; - - if(player.pixel_position < 32) - { - player.pixel_position &= ~(adder - 1); int output_cursor = 0; while(player.pixel_position < 32 && output_cursor < length) { @@ -701,32 +698,29 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, player.pixel_position += adder; } } - - // the next interesting event is after next_event_time cycles, so progress - position = (position + length) % 160; - start = next_event_time; - - // if the event is a motion tick, apply - if(is_moving_[position_identity] && start == next_motion_time) + else { - perform_motion_step(position_identity); - next_motion_time += 4; - } - - // if it's a draw trigger, trigger a draw - if(position == (next_copy % 160)) - { - player.pixel_position = 0; + player.pixel_position = std::max(32, player.pixel_position + length * adder); } } -// } -// else -// { -// // just advance the timer all in one jump -// const int length = end - start; -// position = (position + length) % 160; -// player.pixel_position = std::min(32, player.pixel_position + adder * length); -// } + + // the next interesting event is after next_event_time cycles, so progress + position = (position + length) % 160; + start = next_event_time; + + // if the event is a motion tick, apply + if(is_moving_[position_identity] && start == next_motion_time) + { + perform_motion_step(position_identity); + next_motion_time += 4; + } + + // if it's a draw trigger, trigger a draw + if(start == next_copy_time) + { + player.pixel_position = 0; + } + } } void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) From 99a35266e143c227aad92c0f91f53e07f1fecb16 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 19 Feb 2017 21:20:37 -0500 Subject: [PATCH 60/91] Attempted to bring frequency-switching logic into the cross-platform realm. Which for now creates an issue with the OpenGL context. --- Machines/Atari2600/Atari2600.cpp | 47 ++++++++++++- Machines/Atari2600/Atari2600.hpp | 17 ++++- Machines/Atari2600/TIA.cpp | 66 +++++++++---------- .../Machine/Wrappers/CSAtari2600.mm | 36 ---------- 4 files changed, 95 insertions(+), 71 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index 64ed2e963..d796fd983 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -21,7 +21,9 @@ Machine::Machine() : rom_pages_{nullptr, nullptr, nullptr, nullptr}, tia_input_value_{0xff, 0xff}, cycles_since_speaker_update_(0), - cycles_since_video_update_(0) + cycles_since_video_update_(0), + frame_record_pointer_(0), + is_ntsc_(true) { set_clock_rate(NTSC_clock_rate); } @@ -31,6 +33,7 @@ void Machine::setup_output(float aspect_ratio) tia_.reset(new TIA); speaker_.reset(new Speaker); speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); + tia_->get_crt()->set_delegate(this); } void Machine::close_output() @@ -288,3 +291,45 @@ void Machine::synchronise() update_video(); speaker_->flush(); } + +#pragma mark - CRT delegate + +void Machine::crt_did_end_batch_of_frames(Outputs::CRT::CRT *crt, unsigned int number_of_frames, unsigned int number_of_unexpected_vertical_syncs) +{ + frame_records_[frame_record_pointer_].number_of_frames = number_of_frames; + frame_records_[frame_record_pointer_].number_of_unexpected_vertical_syncs = number_of_unexpected_vertical_syncs; + const size_t number_of_frame_records = sizeof(frame_records_) / sizeof(frame_records_[0]); + frame_record_pointer_ = (frame_record_pointer_ + 1) % number_of_frame_records; + + unsigned int total_number_of_frames = 0; + unsigned int total_number_of_unexpected_vertical_syncs = 0; + for(size_t c = 0; c < number_of_frame_records; c++) + { + if(!frame_records_[c].number_of_frames) return; + total_number_of_frames += frame_records_[c].number_of_frames; + total_number_of_unexpected_vertical_syncs += frame_records_[c].number_of_unexpected_vertical_syncs; + } + + if(total_number_of_unexpected_vertical_syncs >= total_number_of_frames >> 1) + { + for(size_t c = 0; c < number_of_frame_records; c++) + { + frame_records_[c].number_of_frames = 0; + frame_records_[c].number_of_unexpected_vertical_syncs = 0; + } + is_ntsc_ ^= true; + + if(is_ntsc_) + { + set_clock_rate(NTSC_clock_rate); + tia_->set_output_mode(TIA::OutputMode::NTSC); + } + else + { + set_clock_rate(PAL_clock_rate); + tia_->set_output_mode(TIA::OutputMode::PAL); + } + + speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); + } +} diff --git a/Machines/Atari2600/Atari2600.hpp b/Machines/Atari2600/Atari2600.hpp index d78bae584..f0970348e 100644 --- a/Machines/Atari2600/Atari2600.hpp +++ b/Machines/Atari2600/Atari2600.hpp @@ -28,7 +28,8 @@ const unsigned int number_of_recorded_counters = 7; class Machine: public CPU6502::Processor, public CRTMachine::Machine, - public ConfigurationTarget::Machine { + public ConfigurationTarget::Machine, + public Outputs::CRT::Delegate { public: Machine(); @@ -51,6 +52,9 @@ class Machine: virtual std::shared_ptr get_speaker() { return speaker_; } virtual void run_for_cycles(int number_of_cycles) { CPU6502::Processor::run_for_cycles(number_of_cycles); } + // to satisfy Outputs::CRT::Delegate + virtual void crt_did_end_batch_of_frames(Outputs::CRT::CRT *crt, unsigned int number_of_frames, unsigned int number_of_unexpected_vertical_syncs); + private: uint8_t *rom_, *rom_pages_[4]; size_t rom_size_; @@ -72,6 +76,17 @@ class Machine: // video backlog accumulation counter unsigned int cycles_since_video_update_; void update_video(); + + // output frame rate tracker + struct FrameRecord + { + unsigned int number_of_frames; + unsigned int number_of_unexpected_vertical_syncs; + + FrameRecord() : number_of_frames(0), number_of_unexpected_vertical_syncs(0) {} + } frame_records_[4]; + unsigned int frame_record_pointer_; + bool is_ntsc_; }; } diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index fd5ebb3fd..1ba2660b4 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -143,17 +143,40 @@ TIA::TIA(std::function line_end_function) : TIA(fa void TIA::set_output_mode(Atari2600::TIA::OutputMode output_mode) { - // this is the NTSC phase offset function; see below for PAL - crt_->set_composite_sampling_function( - "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" - "{" - "uint c = texture(texID, coordinate).r;" - "uint y = c & 14u;" - "uint iPhase = (c >> 4);" + Outputs::CRT::DisplayType display_type; + + if(output_mode == OutputMode::NTSC) + { + crt_->set_composite_sampling_function( + "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" + "{" + "uint c = texture(texID, coordinate).r;" + "uint y = c & 14u;" + "uint iPhase = (c >> 4);" + + "float phaseOffset = 6.283185308 * float(iPhase) / 13.0 + 5.074880441076923;" + "return mix(float(y) / 14.0, step(1, iPhase) * cos(phase + phaseOffset), amplitude);" + "}"); + display_type = Outputs::CRT::DisplayType::NTSC60; + } + else + { + crt_->set_composite_sampling_function( + "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" + "{" + "uint c = texture(texID, coordinate).r;" + "uint y = c & 14u;" + "uint iPhase = (c >> 4);" + + "uint direction = iPhase & 1u;" + "float phaseOffset = float(7u - direction) + (float(direction) - 0.5) * 2.0 * float(iPhase >> 1);" + "phaseOffset *= 6.283185308 / 12.0;" + "return mix(float(y) / 14.0, step(4, (iPhase + 2u) & 15u) * cos(phase + phaseOffset), amplitude);" + "}"); + display_type = Outputs::CRT::DisplayType::PAL50; + } + crt_->set_new_display_type(cycles_per_line * 2 + 1, display_type); - "float phaseOffset = 6.283185308 * float(iPhase) / 13.0 + 5.074880441076923;" - "return mix(float(y) / 14.0, step(1, iPhase) * cos(phase + phaseOffset), amplitude);" - "}"); /* speaker_->set_input_rate((float)(get_clock_rate() / 38.0));*/ } @@ -161,29 +184,6 @@ TIA::~TIA() { } -/*void Machine::switch_region() -{ - // the PAL function - crt_->set_composite_sampling_function( - "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" - "{" - "uint c = texture(texID, coordinate).r;" - "uint y = c & 14u;" - "uint iPhase = (c >> 4);" - - "uint direction = iPhase & 1u;" - "float phaseOffset = float(7u - direction) + (float(direction) - 0.5) * 2.0 * float(iPhase >> 1);" - "phaseOffset *= 6.283185308 / 12.0;" - "return mix(float(y) / 14.0, step(4, (iPhase + 2u) & 15u) * cos(phase + phaseOffset), amplitude);" - "}"); - - crt_->set_new_timing(228, 312, Outputs::CRT::ColourSpace::YUV, 228, 1, true); - - is_pal_region_ = true; - speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); - set_clock_rate(PAL_clock_rate); -}*/ - // justification for +5: "we need to wait at least 71 [clocks] before the HMOVE operation is complete"; // which will take 16*4 + 2 = 66 cycles from the first compare, implying the first compare must be // in five cycles from now diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm index 099611af8..08c003fbc 100644 --- a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAtari2600.mm @@ -11,42 +11,8 @@ #include "Atari2600.hpp" #import "CSMachine+Subclassing.h" -@interface CSAtari2600 () -- (void)crt:(Outputs::CRT::CRT *)crt didEndBatchOfFrames:(unsigned int)numberOfFrames withUnexpectedVerticalSyncs:(unsigned int)numberOfUnexpectedSyncs; -@end - -struct CRTDelegate: public Outputs::CRT::Delegate { - __weak CSAtari2600 *atari2600; - void crt_did_end_batch_of_frames(Outputs::CRT::CRT *crt, unsigned int number_of_frames, unsigned int number_of_unexpected_vertical_syncs) { - [atari2600 crt:crt didEndBatchOfFrames:number_of_frames withUnexpectedVerticalSyncs:number_of_unexpected_vertical_syncs]; - } -}; - @implementation CSAtari2600 { Atari2600::Machine _atari2600; - CRTDelegate _crtDelegate; - - int _frameCount; - int _hitCount; - BOOL _didDecideRegion; - int _batchesReceived; -} - -- (void)crt:(Outputs::CRT::CRT *)crt didEndBatchOfFrames:(unsigned int)numberOfFrames withUnexpectedVerticalSyncs:(unsigned int)numberOfUnexpectedSyncs { - if(!_didDecideRegion) - { - _batchesReceived++; - if(_batchesReceived == 2) - { - _didDecideRegion = YES; - if(numberOfUnexpectedSyncs >= numberOfFrames >> 1) - { - [self.view performWithGLContext:^{ -// _atari2600.switch_region(); - }]; - } - } - } } - (void)setDirection:(CSJoystickDirection)direction onPad:(NSUInteger)pad isPressed:(BOOL)isPressed { @@ -78,8 +44,6 @@ struct CRTDelegate: public Outputs::CRT::Delegate { - (void)setupOutputWithAspectRatio:(float)aspectRatio { @synchronized(self) { [super setupOutputWithAspectRatio:aspectRatio]; - _atari2600.get_crt()->set_delegate(&_crtDelegate); - _crtDelegate.atari2600 = self; } } From fccdce65b9dd933bf937c60b49ec60013889854f Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 19 Feb 2017 21:45:28 -0500 Subject: [PATCH 61/91] Switched to lock guards. --- Outputs/CRT/Internals/Shaders/Shader.cpp | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/Outputs/CRT/Internals/Shaders/Shader.cpp b/Outputs/CRT/Internals/Shaders/Shader.cpp index 15b8bb43c..4dc5fa90d 100644 --- a/Outputs/CRT/Internals/Shaders/Shader.cpp +++ b/Outputs/CRT/Internals/Shaders/Shader.cpp @@ -292,18 +292,16 @@ void Shader::set_uniform_matrix(const std::string &name, GLint size, GLsizei cou void Shader::enqueue_function(std::function function) { - function_mutex_.lock(); + std::lock_guard function_guard(function_mutex_); enqueued_functions_.push_back(function); - function_mutex_.unlock(); } void Shader::flush_functions() { - function_mutex_.lock(); + std::lock_guard function_guard(function_mutex_); for(std::function function : enqueued_functions_) { function(); } enqueued_functions_.clear(); - function_mutex_.unlock(); } From d979a822ac3dc043d61c2d79da3833c0f9ec0d6e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 19 Feb 2017 21:46:07 -0500 Subject: [PATCH 62/91] Introduced a deferred task list for the OpenGL thread. --- Outputs/CRT/CRT.hpp | 19 ++++++++++++++++++- 1 file changed, 18 insertions(+), 1 deletion(-) diff --git a/Outputs/CRT/CRT.hpp b/Outputs/CRT/CRT.hpp index af06af273..df78efe56 100644 --- a/Outputs/CRT/CRT.hpp +++ b/Outputs/CRT/CRT.hpp @@ -81,10 +81,19 @@ class CRT { uint16_t x1, y; } output_run_; - // The delegate + // the delegate Delegate *delegate_; unsigned int frames_since_last_delegate_call_; + // queued tasks for the OpenGL queue; performed before the next draw + std::mutex function_mutex_; + std::vector> enqueued_openGL_functions_; + inline void enqueue_openGL_function(const std::function &function) + { + std::lock_guard function_guard(function_mutex_); + enqueued_openGL_functions_.push_back(function); + } + public: /*! Constructs the CRT with a specified clock rate, height and colour subcarrier frequency. The requested number of buffers, each with the requested number of bytes per pixel, @@ -204,6 +213,14 @@ class CRT { */ inline void draw_frame(unsigned int output_width, unsigned int output_height, bool only_if_dirty) { + { + std::lock_guard function_guard(function_mutex_); + for(std::function function : enqueued_openGL_functions_) + { + function(); + } + enqueued_openGL_functions_.clear(); + } openGL_output_builder_.draw_frame(output_width, output_height, only_if_dirty); } From 6cb95b4fc540879b62fe139f0fe32a899b135574 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 20 Feb 2017 10:35:33 -0500 Subject: [PATCH 63/91] Switched to passing around `std::string`s rather than `char *`s, because they should be easier to capture. --- Outputs/CRT/CRT.hpp | 4 ++-- Outputs/CRT/Internals/CRTOpenGL.cpp | 19 ++++++--------- Outputs/CRT/Internals/CRTOpenGL.hpp | 8 +++---- .../Internals/Shaders/IntermediateShader.cpp | 20 ++++++++-------- .../Internals/Shaders/IntermediateShader.hpp | 6 ++--- Outputs/CRT/Internals/Shaders/Shader.cpp | 23 ++++++++++--------- Outputs/CRT/Internals/Shaders/Shader.hpp | 4 ++-- 7 files changed, 41 insertions(+), 43 deletions(-) diff --git a/Outputs/CRT/CRT.hpp b/Outputs/CRT/CRT.hpp index df78efe56..f64562800 100644 --- a/Outputs/CRT/CRT.hpp +++ b/Outputs/CRT/CRT.hpp @@ -243,7 +243,7 @@ class CRT { that evaluates to the composite signal level as a function of a source buffer, sampling location, colour carrier phase and amplitude. */ - inline void set_composite_sampling_function(const char *shader) + inline void set_composite_sampling_function(const std::string &shader) { openGL_output_builder_.set_composite_sampling_function(shader); } @@ -261,7 +261,7 @@ class CRT { * `vec2 coordinate` representing the source buffer location to sample from in the range [0, 1); and * `vec2 icoordinate` representing the source buffer location to sample from as a pixel count, for easier multiple-pixels-per-byte unpacking. */ - inline void set_rgb_sampling_function(const char *shader) + inline void set_rgb_sampling_function(const std::string &shader) { openGL_output_builder_.set_rgb_sampling_function(shader); } diff --git a/Outputs/CRT/Internals/CRTOpenGL.cpp b/Outputs/CRT/Internals/CRTOpenGL.cpp index 2b471fa3d..dc5166c9c 100644 --- a/Outputs/CRT/Internals/CRTOpenGL.cpp +++ b/Outputs/CRT/Internals/CRTOpenGL.cpp @@ -8,7 +8,6 @@ #include "CRT.hpp" #include -#include #include #include "CRTOpenGL.hpp" @@ -17,7 +16,8 @@ using namespace Outputs::CRT; -namespace { +namespace +{ static const GLenum source_data_texture_unit = GL_TEXTURE0; static const GLenum pixel_accumulation_texture_unit = GL_TEXTURE1; @@ -31,8 +31,6 @@ namespace { OpenGLOutputBuilder::OpenGLOutputBuilder(size_t bytes_per_pixel) : visible_area_(Rect(0, 0, 1, 1)), composite_src_output_y_(0), - composite_shader_(nullptr), - rgb_shader_(nullptr), last_output_width_(0), last_output_height_(0), fence_(nullptr), @@ -78,9 +76,6 @@ OpenGLOutputBuilder::OpenGLOutputBuilder(size_t bytes_per_pixel) : OpenGLOutputBuilder::~OpenGLOutputBuilder() { glDeleteVertexArrays(1, &output_vertex_array_); - - free(composite_shader_); - free(rgb_shader_); } bool OpenGLOutputBuilder::get_is_television_output() @@ -275,17 +270,17 @@ void OpenGLOutputBuilder::set_openGL_context_will_change(bool should_delete_reso output_mutex_.unlock(); } -void OpenGLOutputBuilder::set_composite_sampling_function(const char *shader) +void OpenGLOutputBuilder::set_composite_sampling_function(const std::string &shader) { std::lock_guard lock_guard(output_mutex_); - composite_shader_ = strdup(shader); + composite_shader_ = shader; reset_all_OpenGL_state(); } -void OpenGLOutputBuilder::set_rgb_sampling_function(const char *shader) +void OpenGLOutputBuilder::set_rgb_sampling_function(const std::string &shader) { std::lock_guard lock_guard(output_mutex_); - rgb_shader_ = strdup(shader); + rgb_shader_ = shader; reset_all_OpenGL_state(); } @@ -321,7 +316,7 @@ void OpenGLOutputBuilder::prepare_composite_input_shaders() void OpenGLOutputBuilder::prepare_rgb_input_shaders() { - if(rgb_shader_) + if(rgb_shader_.size()) { rgb_input_shader_program_ = OpenGL::IntermediateShader::make_rgb_source_shader(rgb_shader_); rgb_input_shader_program_->set_source_texture_unit(source_data_texture_unit); diff --git a/Outputs/CRT/Internals/CRTOpenGL.hpp b/Outputs/CRT/Internals/CRTOpenGL.hpp index b14d49e28..cb980b8fc 100644 --- a/Outputs/CRT/Internals/CRTOpenGL.hpp +++ b/Outputs/CRT/Internals/CRTOpenGL.hpp @@ -47,8 +47,8 @@ class OpenGLOutputBuilder { Rect visible_area_; // Other things the caller may have provided. - char *composite_shader_; - char *rgb_shader_; + std::string composite_shader_; + std::string rgb_shader_; // Methods used by the OpenGL code void prepare_output_shader(); @@ -148,8 +148,8 @@ class OpenGLOutputBuilder { void draw_frame(unsigned int output_width, unsigned int output_height, bool only_if_dirty); void set_openGL_context_will_change(bool should_delete_resources); - void set_composite_sampling_function(const char *shader); - void set_rgb_sampling_function(const char *shader); + void set_composite_sampling_function(const std::string &shader); + void set_rgb_sampling_function(const std::string &shader); void set_output_device(OutputDevice output_device); void set_timing(unsigned int input_frequency, unsigned int cycles_per_line, unsigned int height_of_display, unsigned int horizontal_scan_period, unsigned int vertical_scan_period, unsigned int vertical_period_divider); }; diff --git a/Outputs/CRT/Internals/Shaders/IntermediateShader.cpp b/Outputs/CRT/Internals/Shaders/IntermediateShader.cpp index 870eef24f..40c8bebbe 100644 --- a/Outputs/CRT/Internals/Shaders/IntermediateShader.cpp +++ b/Outputs/CRT/Internals/Shaders/IntermediateShader.cpp @@ -26,7 +26,7 @@ namespace { }; } -std::unique_ptr IntermediateShader::make_shader(const char *fragment_shader, bool use_usampler, bool input_is_inputPosition) +std::unique_ptr IntermediateShader::make_shader(const std::string &fragment_shader, bool use_usampler, bool input_is_inputPosition) { const char *sampler_type = use_usampler ? "usampler2D" : "sampler2D"; const char *input_variable = input_is_inputPosition ? "inputPosition" : "outputPosition"; @@ -111,12 +111,13 @@ std::unique_ptr IntermediateShader::make_shader(const char * return shader; } -std::unique_ptr IntermediateShader::make_source_conversion_shader(const char *composite_shader, const char *rgb_shader) +std::unique_ptr IntermediateShader::make_source_conversion_shader(const std::string &composite_shader, const std::string &rgb_shader) { - char *composite_sample = (char *)composite_shader; - if(!composite_sample) + char *derived_composite_sample = nullptr; + const char *composite_sample = composite_shader.c_str(); + if(!composite_shader.size()) { - asprintf(&composite_sample, + asprintf(&derived_composite_sample, "%s\n" "uniform mat3 rgbToLumaChroma;" "float composite_sample(usampler2D texID, vec2 coordinate, vec2 iCoordinate, float phase, float amplitude)" @@ -126,7 +127,8 @@ std::unique_ptr IntermediateShader::make_source_conversion_s "vec2 quadrature = vec2(cos(phase), -sin(phase)) * amplitude;" "return dot(lumaChromaColour, vec3(1.0 - amplitude, quadrature));" "}", - rgb_shader); + rgb_shader.c_str()); + composite_sample = derived_composite_sample; } char *fragment_shader; @@ -148,7 +150,7 @@ std::unique_ptr IntermediateShader::make_source_conversion_s "fragColour = vec4(composite_sample(texID, inputPositionsVarying[5], iInputPositionVarying, phaseAndAmplitudeVarying.x, phaseAndAmplitudeVarying.y));" "}" , composite_sample); - if(!composite_shader) free(composite_sample); + free(derived_composite_sample); std::unique_ptr shader = make_shader(fragment_shader, true, true); free(fragment_shader); @@ -156,7 +158,7 @@ std::unique_ptr IntermediateShader::make_source_conversion_s return shader; } -std::unique_ptr IntermediateShader::make_rgb_source_shader(const char *rgb_shader) +std::unique_ptr IntermediateShader::make_rgb_source_shader(const std::string &rgb_shader) { char *fragment_shader; asprintf(&fragment_shader, @@ -176,7 +178,7 @@ std::unique_ptr IntermediateShader::make_rgb_source_shader(c "{" "fragColour = rgb_sample(texID, inputPositionsVarying[5], iInputPositionVarying);" "}" - , rgb_shader); + , rgb_shader.c_str()); std::unique_ptr shader = make_shader(fragment_shader, true, true); free(fragment_shader); diff --git a/Outputs/CRT/Internals/Shaders/IntermediateShader.hpp b/Outputs/CRT/Internals/Shaders/IntermediateShader.hpp index 13dedc2c4..fc79e5fd7 100644 --- a/Outputs/CRT/Internals/Shaders/IntermediateShader.hpp +++ b/Outputs/CRT/Internals/Shaders/IntermediateShader.hpp @@ -25,13 +25,13 @@ public: converting them to single-channel composite values using @c composite_shader if supplied or @c rgb_shader and a reference composite conversion if @c composite_shader is @c nullptr. */ - static std::unique_ptr make_source_conversion_shader(const char *composite_shader, const char *rgb_shader); + static std::unique_ptr make_source_conversion_shader(const std::string &composite_shader, const std::string &rgb_shader); /*! Constructs and returns an intermediate shader that will take runs from the inputPositions, converting them to RGB values using @c rgb_shader. */ - static std::unique_ptr make_rgb_source_shader(const char *rgb_shader); + static std::unique_ptr make_rgb_source_shader(const std::string &rgb_shader); /*! Constructs and returns an intermediate shader that will read composite samples from the R channel, @@ -95,7 +95,7 @@ public: void set_is_double_height(bool is_double_height, float input_offset = 0.0f, float output_offset = 0.0f); private: - static std::unique_ptr make_shader(const char *fragment_shader, bool use_usampler, bool input_is_inputPosition); + static std::unique_ptr make_shader(const std::string &fragment_shader, bool use_usampler, bool input_is_inputPosition); }; } diff --git a/Outputs/CRT/Internals/Shaders/Shader.cpp b/Outputs/CRT/Internals/Shaders/Shader.cpp index 4dc5fa90d..ddc723c4a 100644 --- a/Outputs/CRT/Internals/Shaders/Shader.cpp +++ b/Outputs/CRT/Internals/Shaders/Shader.cpp @@ -8,22 +8,23 @@ #include "Shader.hpp" -#include #include using namespace OpenGL; -namespace { +namespace +{ Shader *bound_shader = nullptr; } -GLuint Shader::compile_shader(const char *source, GLenum type) +GLuint Shader::compile_shader(const std::string &source, GLenum type) { GLuint shader = glCreateShader(type); - glShaderSource(shader, 1, &source, NULL); + const char *c_str = source.c_str(); + glShaderSource(shader, 1, &c_str, NULL); glCompileShader(shader); -#if defined(DEBUG) +#ifdef DEBUG GLint isCompiled = 0; glGetShaderiv(shader, GL_COMPILE_STATUS, &isCompiled); if(isCompiled == GL_FALSE) @@ -31,10 +32,10 @@ GLuint Shader::compile_shader(const char *source, GLenum type) GLint logLength; glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &logLength); if(logLength > 0) { - GLchar *log = (GLchar *)malloc((size_t)logLength); + GLchar *log = new GLchar[logLength]; glGetShaderInfoLog(shader, logLength, &logLength, log); printf("Compile log:\n%s\n", log); - free(log); + delete[] log; } throw (type == GL_VERTEX_SHADER) ? VertexShaderCompilationError : FragmentShaderCompilationError; @@ -44,7 +45,7 @@ GLuint Shader::compile_shader(const char *source, GLenum type) return shader; } -Shader::Shader(const char *vertex_shader, const char *fragment_shader, const AttributeBinding *attribute_bindings) +Shader::Shader(const std::string &vertex_shader, const std::string &fragment_shader, const AttributeBinding *attribute_bindings) { shader_program_ = glCreateProgram(); GLuint vertex = compile_shader(vertex_shader, GL_VERTEX_SHADER); @@ -64,7 +65,7 @@ Shader::Shader(const char *vertex_shader, const char *fragment_shader, const Att glLinkProgram(shader_program_); -#if defined(DEBUG) +#ifdef DEBUG GLint didLink = 0; glGetProgramiv(shader_program_, GL_LINK_STATUS, &didLink); if(didLink == GL_FALSE) @@ -72,10 +73,10 @@ Shader::Shader(const char *vertex_shader, const char *fragment_shader, const Att GLint logLength; glGetProgramiv(shader_program_, GL_INFO_LOG_LENGTH, &logLength); if(logLength > 0) { - GLchar *log = (GLchar *)malloc((size_t)logLength); + GLchar *log = new GLchar[logLength]; glGetProgramInfoLog(shader_program_, logLength, &logLength, log); printf("Link log:\n%s\n", log); - free(log); + delete[] log; } throw ProgramLinkageError; } diff --git a/Outputs/CRT/Internals/Shaders/Shader.hpp b/Outputs/CRT/Internals/Shaders/Shader.hpp index ba95daf7c..f2afa3ab9 100644 --- a/Outputs/CRT/Internals/Shaders/Shader.hpp +++ b/Outputs/CRT/Internals/Shaders/Shader.hpp @@ -41,7 +41,7 @@ public: @param fragment_shader The fragment shader source code. @param attribute_bindings Either @c nullptr or an array terminated by an entry with a @c nullptr-name of attribute bindings. */ - Shader(const char *vertex_shader, const char *fragment_shader, const AttributeBinding *attribute_bindings); + Shader(const std::string &vertex_shader, const std::string &fragment_shader, const AttributeBinding *attribute_bindings); ~Shader(); /*! @@ -106,7 +106,7 @@ public: void set_uniform_matrix(const std::string &name, GLint size, GLsizei count, bool transpose, const GLfloat *values); private: - GLuint compile_shader(const char *source, GLenum type); + GLuint compile_shader(const std::string &source, GLenum type); GLuint shader_program_; void flush_functions(); From d19f26887d6de1922cc37fe62f62b27d977c67d5 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 20 Feb 2017 10:39:31 -0500 Subject: [PATCH 64/91] Performed a very naive shuffling of output builder sets onto the OpenGL queue. Which makes the frequency switcher work properly from it's possibly-contextless thread. --- Outputs/CRT/CRT.hpp | 20 +++++++++++++++----- 1 file changed, 15 insertions(+), 5 deletions(-) diff --git a/Outputs/CRT/CRT.hpp b/Outputs/CRT/CRT.hpp index f64562800..559c69a3e 100644 --- a/Outputs/CRT/CRT.hpp +++ b/Outputs/CRT/CRT.hpp @@ -233,7 +233,9 @@ class CRT { */ inline void set_openGL_context_will_change(bool should_delete_resources) { - openGL_output_builder_.set_openGL_context_will_change(should_delete_resources); + enqueue_openGL_function([should_delete_resources, this] { + openGL_output_builder_.set_openGL_context_will_change(should_delete_resources); + }); } /*! Sets a function that will map from whatever data the machine provided to a composite signal. @@ -245,7 +247,9 @@ class CRT { */ inline void set_composite_sampling_function(const std::string &shader) { - openGL_output_builder_.set_composite_sampling_function(shader); + enqueue_openGL_function([shader, this] { + openGL_output_builder_.set_composite_sampling_function(shader); + }); } /*! Sets a function that will map from whatever data the machine provided to an RGB signal. @@ -263,17 +267,23 @@ class CRT { */ inline void set_rgb_sampling_function(const std::string &shader) { - openGL_output_builder_.set_rgb_sampling_function(shader); + enqueue_openGL_function([shader, this] { + openGL_output_builder_.set_rgb_sampling_function(shader); + }); } inline void set_output_device(OutputDevice output_device) { - openGL_output_builder_.set_output_device(output_device); + enqueue_openGL_function([output_device, this] { + openGL_output_builder_.set_output_device(output_device); + }); } inline void set_visible_area(Rect visible_area) { - openGL_output_builder_.set_visible_area(visible_area); + enqueue_openGL_function([visible_area, this] { + openGL_output_builder_.set_visible_area(visible_area); + }); } Rect get_rect_for_area(int first_line_after_sync, int number_of_lines, int first_cycle_after_sync, int number_of_cycles, float aspect_ratio); From 87afa9140ee5c74abb656fa4419e4cc590bbec82 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 20 Feb 2017 17:44:36 -0500 Subject: [PATCH 65/91] Took some provision steps towards paging type autodetection and communication. But I think this is a distraction. --- StaticAnalyser/Atari/StaticAnalyser.cpp | 28 +++++++++++++++++++ .../Disassembler/Disassembler6502.cpp | 22 +++++++-------- .../Disassembler/Disassembler6502.hpp | 2 +- StaticAnalyser/StaticAnalyser.hpp | 24 ++++++++++++---- 4 files changed, 59 insertions(+), 17 deletions(-) diff --git a/StaticAnalyser/Atari/StaticAnalyser.cpp b/StaticAnalyser/Atari/StaticAnalyser.cpp index 231b40e6b..abce4327d 100644 --- a/StaticAnalyser/Atari/StaticAnalyser.cpp +++ b/StaticAnalyser/Atari/StaticAnalyser.cpp @@ -8,6 +8,8 @@ #include "StaticAnalyser.hpp" +#include "../Disassembler/Disassembler6502.hpp" + using namespace StaticAnalyser::Atari; void StaticAnalyser::Atari::AddTargets( @@ -24,5 +26,31 @@ void StaticAnalyser::Atari::AddTargets( target.disks = disks; target.tapes = tapes; target.cartridges = cartridges; + target.atari.paging_model = Atari2600PagingModel::None; + + // try to figure out the paging scheme +/* if(!cartridges.empty()) + { + const std::list &segments = cartridges.front()->get_segments(); + if(segments.size() == 1) + { + uint16_t entry_address, break_address; + const Storage::Cartridge::Cartridge::Segment &segment = segments.front(); + if(segment.data.size() < 4096) + { + entry_address = (uint16_t)(segment.data[0x7fc] | (segment.data[0x7fd] << 8)); + break_address = (uint16_t)(segment.data[0x7fe] | (segment.data[0x7ff] << 8)); + } + else + { + entry_address = (uint16_t)(segment.data[0xffc] | (segment.data[0xffd] << 8)); + break_address = (uint16_t)(segment.data[0xffe] | (segment.data[0xfff] << 8)); + } + StaticAnalyser::MOS6502::Disassembly disassembly = + StaticAnalyser::MOS6502::Disassemble(segment.data, 0x1000, {entry_address, break_address}, 0x1fff); + printf("%p", &disassembly); + } + }*/ + destination.push_back(target); } diff --git a/StaticAnalyser/Disassembler/Disassembler6502.cpp b/StaticAnalyser/Disassembler/Disassembler6502.cpp index 3b773f10e..bb50ddf33 100644 --- a/StaticAnalyser/Disassembler/Disassembler6502.cpp +++ b/StaticAnalyser/Disassembler/Disassembler6502.cpp @@ -16,9 +16,9 @@ struct PartialDisassembly { std::vector remaining_entry_points; }; -static void AddToDisassembly(PartialDisassembly &disassembly, const std::vector &memory, uint16_t start_address, uint16_t entry_point) +static void AddToDisassembly(PartialDisassembly &disassembly, const std::vector &memory, uint16_t start_address, uint16_t entry_point, uint16_t address_mask) { - uint16_t address = entry_point; + uint16_t address = entry_point & address_mask; while(1) { uint16_t local_address = address - start_address; @@ -26,7 +26,7 @@ static void AddToDisassembly(PartialDisassembly &disassembly, const std::vector< struct Instruction instruction; instruction.address = address; - address++; + address = (address + 1) & address_mask; // get operation uint8_t operation = memory[local_address]; @@ -250,7 +250,7 @@ static void AddToDisassembly(PartialDisassembly &disassembly, const std::vector< if(operand_address >= memory.size()-1) return; address += 2; - instruction.operand = memory[operand_address] | (uint16_t)(memory[operand_address+1] << 8); + instruction.operand = memory[operand_address] | (uint16_t)(memory[operand_address + 1] << 8); } break; } @@ -259,7 +259,7 @@ static void AddToDisassembly(PartialDisassembly &disassembly, const std::vector< disassembly.disassembly.instructions_by_address[instruction.address] = instruction; // TODO: something wider-ranging than this - if(instruction.addressing_mode == Instruction::Absolute && (instruction.operand < start_address || instruction.operand >= start_address + memory.size())) + if((instruction.addressing_mode == Instruction::Absolute || instruction.addressing_mode == Instruction::ZeroPage) && (instruction.operand < start_address || instruction.operand >= start_address + memory.size())) { if( instruction.operation == Instruction::STY || instruction.operation == Instruction::STX || @@ -276,23 +276,23 @@ static void AddToDisassembly(PartialDisassembly &disassembly, const std::vector< if(instruction.operation == Instruction::BRK) return; // TODO: check whether IRQ vector is within memory range if(instruction.operation == Instruction::JSR) { - disassembly.remaining_entry_points.push_back(instruction.operand); + disassembly.remaining_entry_points.push_back(instruction.operand & address_mask); } if(instruction.operation == Instruction::JMP) { if(instruction.addressing_mode == Instruction::Absolute) - disassembly.remaining_entry_points.push_back(instruction.operand); + disassembly.remaining_entry_points.push_back(instruction.operand & address_mask); return; } if(instruction.addressing_mode == Instruction::Relative) { uint16_t destination = (uint16_t)(address + (int8_t)instruction.operand); - disassembly.remaining_entry_points.push_back(destination); + disassembly.remaining_entry_points.push_back(destination & address_mask); } } } -Disassembly StaticAnalyser::MOS6502::Disassemble(const std::vector &memory, uint16_t start_address, std::vector entry_points) +Disassembly StaticAnalyser::MOS6502::Disassemble(const std::vector &memory, uint16_t start_address, std::vector entry_points, uint16_t address_mask) { PartialDisassembly partialDisassembly; partialDisassembly.remaining_entry_points = entry_points; @@ -300,7 +300,7 @@ Disassembly StaticAnalyser::MOS6502::Disassemble(const std::vector &mem while(!partialDisassembly.remaining_entry_points.empty()) { // pull the next entry point from the back of the vector - uint16_t next_entry_point = partialDisassembly.remaining_entry_points.back(); + uint16_t next_entry_point = partialDisassembly.remaining_entry_points.back() & address_mask; partialDisassembly.remaining_entry_points.pop_back(); // if that address has already bene visited, forget about it @@ -310,7 +310,7 @@ Disassembly StaticAnalyser::MOS6502::Disassemble(const std::vector &mem if(next_entry_point < start_address || next_entry_point >= start_address + memory.size()) partialDisassembly.disassembly.outward_calls.insert(next_entry_point); else - AddToDisassembly(partialDisassembly, memory, start_address, next_entry_point); + AddToDisassembly(partialDisassembly, memory, start_address, next_entry_point, address_mask); } return std::move(partialDisassembly.disassembly); diff --git a/StaticAnalyser/Disassembler/Disassembler6502.hpp b/StaticAnalyser/Disassembler/Disassembler6502.hpp index 139542777..aff96ba63 100644 --- a/StaticAnalyser/Disassembler/Disassembler6502.hpp +++ b/StaticAnalyser/Disassembler/Disassembler6502.hpp @@ -67,7 +67,7 @@ struct Disassembly { std::set external_stores, external_loads, external_modifies; }; -Disassembly Disassemble(const std::vector &memory, uint16_t start_address, std::vector entry_points); +Disassembly Disassemble(const std::vector &memory, uint16_t start_address, std::vector entry_points, uint16_t address_mask = 0xffff); } } diff --git a/StaticAnalyser/StaticAnalyser.hpp b/StaticAnalyser/StaticAnalyser.hpp index ffef83a23..1aabae662 100644 --- a/StaticAnalyser/StaticAnalyser.hpp +++ b/StaticAnalyser/StaticAnalyser.hpp @@ -25,6 +25,16 @@ enum class Vic20MemoryModel { ThirtyTwoKB }; +enum class Atari2600PagingModel { + None, + Atari8k, + Atari16k, + Atari32k, + ActivisionStack, + ParkerBros, + Tigervision +}; + /*! A list of disks, tapes and cartridges plus information about the machine to which to attach them and its configuration, and instructions on how to launch the software attached, plus a measure of confidence in this target's correctness. @@ -39,21 +49,25 @@ struct Target { float probability; union { - struct { - Vic20MemoryModel memory_model; - bool has_c1540; - } vic20; - struct { bool has_adfs; bool has_dfs; bool should_shift_restart; } acorn; + struct { + Atari2600PagingModel paging_model; + } atari; + struct { bool use_atmos_rom; bool has_microdisc; } oric; + + struct { + Vic20MemoryModel memory_model; + bool has_c1540; + } vic20; }; std::string loadingCommand; From 57f434c19902390fb995bf894a3d81adfb058e56 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 20 Feb 2017 17:58:28 -0500 Subject: [PATCH 66/91] Reorganised state, with an eye towards unifying object motion and triggers. --- Machines/Atari2600/TIA.cpp | 75 +++++++++++++++++++------------------- Machines/Atari2600/TIA.hpp | 26 ++++++++++--- 2 files changed, 58 insertions(+), 43 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 1ba2660b4..e7a40f370 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -26,9 +26,6 @@ TIA::TIA(bool create_crt) : pixel_target_(nullptr), background_{0, 0}, background_half_mask_(0), - position_{0, 0, 0, 0, 0}, - motion_{0, 0, 0, 0, 0}, - is_moving_{false, false, false, false, false}, horizontal_blank_extend_(false), collision_flags_(0) { @@ -330,12 +327,12 @@ void TIA::set_player_position(int player) // one behind its real hardware value, creating the extra delay; and (ii) the player // code is written to start a draw upon wraparound from 159 to 0, so -1 is the // correct option rather than 159. - position_[(int)MotionIndex::Player0 + player] = -1; + object_[(int)MotionIndex::Player0 + player].position = -1; } void TIA::set_player_motion(int player, uint8_t motion) { - motion_[(int)MotionIndex::Player0 + player] = (motion >> 4)&0xf; + object_[(int)MotionIndex::Player0 + player].motion = (motion >> 4)&0xf; } void TIA::set_player_missile_colour(int player, uint8_t colour) @@ -371,7 +368,7 @@ void TIA::set_ball_delay(bool delay) void TIA::set_ball_position() { - position_[(int)MotionIndex::Ball] = 0; + object_[(int)MotionIndex::Ball].position = 0; // setting the ball position also triggers a draw ball_.pixel_position = ball_.size; @@ -379,20 +376,20 @@ void TIA::set_ball_position() void TIA::set_ball_motion(uint8_t motion) { - motion_[(int)MotionIndex::Ball] = (motion >> 4) & 0xf; + object_[(int)MotionIndex::Ball].motion = (motion >> 4) & 0xf; } void TIA::move() { horizontal_blank_extend_ = true; - is_moving_[0] = is_moving_[1] = is_moving_[2] = is_moving_[3] = is_moving_[4] = true; - motion_step_[0] = motion_step_[1] = motion_step_[2] = motion_step_[3] = motion_step_[4] = 15; - motion_time_[0] = motion_time_[1] = motion_time_[2] = motion_time_[3] = motion_time_[4] = (horizontal_counter_ + 3) & ~3; + object_[0].is_moving = object_[1].is_moving = object_[2].is_moving = object_[3].is_moving = object_[4].is_moving = true; + object_[0].motion_step = object_[1].motion_step = object_[2].motion_step = object_[3].motion_step = object_[4].motion_step = 15; + object_[0].motion_time = object_[1].motion_time = object_[2].motion_time = object_[3].motion_time = object_[4].motion_time = (horizontal_counter_ + 3) & ~3; } void TIA::clear_motion() { - motion_[0] = motion_[1] = motion_[2] = motion_[3] = motion_[4] = 0; + object_[0].motion = object_[1].motion = object_[2].motion = object_[3].motion = object_[4].motion = 0; } uint8_t TIA::get_collision_flags(int offset) @@ -429,7 +426,7 @@ void TIA::output_for_cycles(int number_of_cycles) if(line_end_function_) line_end_function_(collision_buffer_.data()); memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_) horizontal_blank_extend_ = false; - for(int c = 0; c < 5; c++) motion_time_[c] %= 228; + for(int c = 0; c < 5; c++) object_[c].motion_time %= 228; } // accumulate an OR'd version of the output into the collision buffer @@ -621,27 +618,29 @@ void TIA::draw_playfield(int start, int end) void TIA::perform_motion_step(int identity) { - if((motion_step_[identity] ^ (motion_[identity] ^ 8)) == 0xf) - is_moving_[identity] = false; + Object &object = object_[identity]; + if((object.motion_step ^ (object.motion ^ 8)) == 0xf) + object.is_moving = false; else { - position_[identity] ++; - motion_step_[identity] --; - motion_time_[identity] += 4; + object.position ++; + object.motion_step --; + object.motion_time += 4; } } int TIA::perform_border_motion(int identity, int start, int end) { - if(!is_moving_[identity]) return 0; + Object &object = object_[identity]; + if(!object.is_moving) return 0; int steps_taken = 0; - while(is_moving_[identity] && motion_time_[identity] < end) + while(object.is_moving && object.motion_time < end) { perform_motion_step(identity); steps_taken++; } - position_[identity] %= 160; + object.position %= 160; return steps_taken; } @@ -650,35 +649,35 @@ int TIA::perform_border_motion(int identity, int start, int end) void TIA::draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) { - int &position = position_[position_identity]; + Object &object = object_[position_identity]; int adder = 4 >> player.size; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - int next_motion_time = motion_time_[position_identity] - first_pixel_cycle + 4; + int next_motion_time = object.motion_time - first_pixel_cycle + 4; while(start < end) { int next_event_time = end; // is the next event a movement tick? - if(is_moving_[position_identity] && next_motion_time < next_event_time) + if(object.is_moving && next_motion_time < next_event_time) { next_event_time = next_motion_time; } // is the next event a graphics trigger? int next_copy = 160; - if(position < 16 && player.copy_flags&1) + if(object.position < 16 && player.copy_flags&1) { next_copy = 16; - } else if(position < 32 && player.copy_flags&2) + } else if(object.position < 32 && player.copy_flags&2) { next_copy = 32; - } else if(position < 64 && player.copy_flags&4) + } else if(object.position < 64 && player.copy_flags&4) { next_copy = 64; } - int next_copy_time = start + next_copy - position; + int next_copy_time = start + next_copy - object.position; if(next_copy_time < next_event_time) next_event_time = next_copy_time; // the decision is to progress by length @@ -705,11 +704,11 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, } // the next interesting event is after next_event_time cycles, so progress - position = (position + length) % 160; + object.position = (object.position + length) % 160; start = next_event_time; // if the event is a motion tick, apply - if(is_moving_[position_identity] && start == next_motion_time) + if(object.is_moving && start == next_motion_time) { perform_motion_step(position_identity); next_motion_time += 4; @@ -746,7 +745,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in } // move further if required - if(is_moving_[position_identity] && end >= 224 && motion_time_[position_identity] < end) + if(object_[position_identity].is_moving && end >= 224 && object_[position_identity].motion_time < end) { perform_motion_step(position_identity); player.pixel_position = std::min(32, player.pixel_position + adder); @@ -787,7 +786,7 @@ void TIA::draw_ball(int start, int end) } // move further if required - if(is_moving_[(int)MotionIndex::Ball] && end >= 224 && motion_time_[(int)MotionIndex::Ball] < end) + if(object_[(int)MotionIndex::Ball].is_moving && end >= 224 && object_[(int)MotionIndex::Ball].motion_time < end) { perform_motion_step((int)MotionIndex::Ball); ball_.pixel_position = std::max(0, ball_.pixel_position - 1); @@ -796,16 +795,16 @@ void TIA::draw_ball(int start, int end) void TIA::draw_ball_visible(int start, int end) { - int &position = position_[(int)MotionIndex::Ball]; + Object &object = object_[(int)MotionIndex::Ball]; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - int next_motion_time = motion_time_[(int)MotionIndex::Ball] - first_pixel_cycle + 4; + int next_motion_time = object.motion_time - first_pixel_cycle + 4; while(start < end) { int next_event_time = end; // is the next event a movement tick? - if(is_moving_[(int)MotionIndex::Ball] && next_motion_time < next_event_time) + if(object.is_moving && next_motion_time < next_event_time) { next_event_time = next_motion_time; } @@ -813,7 +812,7 @@ void TIA::draw_ball_visible(int start, int end) // is the next event a graphics trigger? if(ball_.enabled[ball_.enabled_index]) { - int time_until_copy = 160 - position; + int time_until_copy = 160 - object.position; int next_copy_time = start + time_until_copy; if(next_copy_time < next_event_time) next_event_time = next_copy_time; } @@ -840,18 +839,18 @@ void TIA::draw_ball_visible(int start, int end) } // the next interesting event is after next_event_time cycles, so progress - position = (position + length) % 160; + object.position = (object.position + length) % 160; start = next_event_time; // if the event is a motion tick, apply - if(is_moving_[(int)MotionIndex::Ball] && start == next_motion_time) + if(object.is_moving && start == next_motion_time) { perform_motion_step((int)MotionIndex::Ball); next_motion_time += 4; } // if it's a draw trigger, trigger a draw - if(!position) + if(!object.position) { ball_.pixel_position = ball_.size; } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 2bb5af4bc..ddf8792d7 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -166,11 +166,27 @@ class TIA { // movement bool horizontal_blank_extend_; - int motion_[5]; - int motion_step_[5]; - int motion_time_[5]; - int position_[5]; - bool is_moving_[5]; + struct Object { + // the two programmer-set values + int position; + int motion; + + // motion_step_ is the current motion counter value; motion_time_ is the next time it will fire + int motion_step; + int motion_time; + + // indicates whether this object is currently undergoing motion + bool is_moving; + + // receives a list of drawing events; 20 is a deliberately over-specified quantity + struct DrawingEvent { + int time; + int copy; + } drawing_events[20]; + int draw_event_read_pointer, draw_event_write_pointer; + + Object() : draw_event_read_pointer(0), draw_event_write_pointer(0), is_moving(false) {}; + } object_[5]; enum class MotionIndex : uint8_t { Ball, Player0, From 2bf784535c4920662634e92c322b2b8cb311c72b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 20 Feb 2017 18:04:40 -0500 Subject: [PATCH 67/91] Simplified calllng. --- Machines/Atari2600/TIA.cpp | 51 +++++++++++++++++--------------------- Machines/Atari2600/TIA.hpp | 20 +++++++-------- 2 files changed, 33 insertions(+), 38 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index e7a40f370..20c88a144 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -433,11 +433,11 @@ void TIA::output_for_cycles(int number_of_cycles) int latent_start = output_cursor + 4; int latent_end = horizontal_counter_ + 4; draw_playfield(latent_start, latent_end); - draw_player(player_[0], CollisionType::Player0, (int)MotionIndex::Player0, output_cursor, horizontal_counter_); - draw_player(player_[1], CollisionType::Player1, (int)MotionIndex::Player1, output_cursor, horizontal_counter_); - draw_missile(missile_[0], CollisionType::Missile0, (int)MotionIndex::Missile0, output_cursor, horizontal_counter_); - draw_missile(missile_[1], CollisionType::Missile1, (int)MotionIndex::Missile1, output_cursor, horizontal_counter_); - draw_ball(output_cursor, horizontal_counter_); + draw_player(player_[0], object_[(int)MotionIndex::Player0], CollisionType::Player0, output_cursor, horizontal_counter_); + draw_player(player_[1], object_[(int)MotionIndex::Player1], CollisionType::Player1, output_cursor, horizontal_counter_); + draw_missile(missile_[0], object_[(int)MotionIndex::Missile0], CollisionType::Missile0, output_cursor, horizontal_counter_); + draw_missile(missile_[1], object_[(int)MotionIndex::Missile1], CollisionType::Missile1, output_cursor, horizontal_counter_); + draw_ball(object_[(int)MotionIndex::Ball], output_cursor, horizontal_counter_); // convert to television signals @@ -616,9 +616,8 @@ void TIA::draw_playfield(int start, int end) #pragma mark - Motion -void TIA::perform_motion_step(int identity) +void TIA::perform_motion_step(Object &object) { - Object &object = object_[identity]; if((object.motion_step ^ (object.motion ^ 8)) == 0xf) object.is_moving = false; else @@ -629,15 +628,14 @@ void TIA::perform_motion_step(int identity) } } -int TIA::perform_border_motion(int identity, int start, int end) +int TIA::perform_border_motion(Object &object, int start, int end) { - Object &object = object_[identity]; if(!object.is_moving) return 0; int steps_taken = 0; while(object.is_moving && object.motion_time < end) { - perform_motion_step(identity); + perform_motion_step(object); steps_taken++; } object.position %= 160; @@ -647,9 +645,8 @@ int TIA::perform_border_motion(int identity, int start, int end) #pragma mark - Player output -void TIA::draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) +void TIA::draw_player_visible(Player &player, Object &object, CollisionType collision_identity, int start, int end) { - Object &object = object_[position_identity]; int adder = 4 >> player.size; // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion @@ -710,7 +707,7 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, // if the event is a motion tick, apply if(object.is_moving && start == next_motion_time) { - perform_motion_step(position_identity); + perform_motion_step(object); next_motion_time += 4; } @@ -722,7 +719,7 @@ void TIA::draw_player_visible(Player &player, CollisionType collision_identity, } } -void TIA::draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end) +void TIA::draw_player(Player &player, Object &object, CollisionType collision_identity, int start, int end) { int adder = 4 >> player.size; int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); @@ -730,7 +727,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // movement works across the entire screen, so do work that falls outside of the pixel area if(start < first_pixel) { - player.pixel_position = std::min(32, player.pixel_position + adder * perform_border_motion(position_identity, start, std::max(end, first_pixel))); + player.pixel_position = std::min(32, player.pixel_position + adder * perform_border_motion(object, start, std::max(end, first_pixel))); } // don't continue to do any drawing if this window ends too early @@ -741,37 +738,37 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in // perform the visible part of the line, if any if(start < 224) { - draw_player_visible(player, collision_identity, position_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); + draw_player_visible(player, object, collision_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); } // move further if required - if(object_[position_identity].is_moving && end >= 224 && object_[position_identity].motion_time < end) + if(object.is_moving && end >= 224 && object.motion_time < end) { - perform_motion_step(position_identity); + perform_motion_step(object); player.pixel_position = std::min(32, player.pixel_position + adder); } } #pragma mark - Missile output -void TIA::draw_missile(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end) +void TIA::draw_missile(Missile &missile, Object &object, CollisionType collision_identity, int start, int end) { } -void TIA::draw_missile_visible(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end) +void TIA::draw_missile_visible(Missile &missile, Object &object, CollisionType collision_identity, int start, int end) { } #pragma mark - Ball output -void TIA::draw_ball(int start, int end) +void TIA::draw_ball(Object &object, int start, int end) { int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); // movement works across the entire screen, so do work that falls outside of the pixel area if(start < first_pixel) { - ball_.pixel_position = std::max(0, ball_.pixel_position - perform_border_motion((int)MotionIndex::Ball, start, std::max(end, first_pixel))); + ball_.pixel_position = std::max(0, ball_.pixel_position - perform_border_motion(object, start, std::max(end, first_pixel))); } // don't continue to do any drawing if this window ends too early @@ -782,21 +779,19 @@ void TIA::draw_ball(int start, int end) // perform the visible part of the line, if any if(start < 224) { - draw_ball_visible(start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); + draw_ball_visible(object, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); } // move further if required if(object_[(int)MotionIndex::Ball].is_moving && end >= 224 && object_[(int)MotionIndex::Ball].motion_time < end) { - perform_motion_step((int)MotionIndex::Ball); + perform_motion_step(object); ball_.pixel_position = std::max(0, ball_.pixel_position - 1); } } -void TIA::draw_ball_visible(int start, int end) +void TIA::draw_ball_visible(Object &object, int start, int end) { - Object &object = object_[(int)MotionIndex::Ball]; - // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion int next_motion_time = object.motion_time - first_pixel_cycle + 4; while(start < end) @@ -845,7 +840,7 @@ void TIA::draw_ball_visible(int start, int end) // if the event is a motion tick, apply if(object.is_moving && start == next_motion_time) { - perform_motion_step((int)MotionIndex::Ball); + perform_motion_step(object); next_motion_time += 4; } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index ddf8792d7..fc59a98b0 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -194,8 +194,10 @@ class TIA { Missile0, Missile1 }; - inline int perform_border_motion(int identity, int start, int end); - inline void perform_motion_step(int identity); + + // motion + inline int perform_border_motion(Object &object, int start, int end); + inline void perform_motion_step(Object &object); // drawing methods and state inline void output_for_cycles(int number_of_cycles); @@ -203,16 +205,14 @@ class TIA { inline void draw_playfield(int start, int end); - inline void draw_player(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); - inline void draw_player_visible(Player &player, CollisionType collision_identity, const int position_identity, int start, int end); + inline void draw_player(Player &player, Object &object, CollisionType collision_identity, int start, int end); + inline void draw_player_visible(Player &player, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_missile(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end); - inline void draw_missile_visible(Missile &missile, CollisionType collision_identity, const int position_identity, int start, int end); + inline void draw_missile(Missile &missile, Object &object, CollisionType collision_identity, int start, int end); + inline void draw_missile_visible(Missile &missile, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_ball(int start, int end); - inline void draw_ball_visible(int start, int end); - - inline void update_motion(int start, int end); + inline void draw_ball(Object &object, int start, int end); + inline void draw_ball_visible(Object &object, int start, int end); int pixels_start_location_; uint8_t *pixel_target_; From 99547181f11c4bee360d58c58c9f997866eac3a5 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 20 Feb 2017 21:42:59 -0500 Subject: [PATCH 68/91] Attempted to template this thing. Without yet a plan in place for pixel lookup timing. --- Machines/Atari2600/TIA.cpp | 86 ++++++++++++++++++++++++++++++++++++++ Machines/Atari2600/TIA.hpp | 12 ++---- 2 files changed, 90 insertions(+), 8 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 20c88a144..23cc6f698 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -643,6 +643,92 @@ int TIA::perform_border_motion(Object &object, int start, int end) return steps_taken; } +template void TIA::draw_object(T &target, Object &object, int start, int end) +{ + int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); + + // movement works across the entire screen, so do work that falls outside of the pixel area + if(start < first_pixel) + { + target.skip_pixels(perform_border_motion(object, start, std::max(end, first_pixel))); + } + + // don't continue to do any drawing if this window ends too early + if(end < first_pixel) return; + if(start < first_pixel) start = first_pixel; + if(start >= end) return; + + // perform the visible part of the line, if any + if(start < 224) + { + draw_object_visible(target, object, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); + } + + // move further if required + if(object.is_moving && end >= 224 && object.motion_time < end) + { + perform_motion_step(object); + target.skip_pixels(1); + } +} + +template void TIA::draw_object_visible(T &target, Object &object, int start, int end) +{ + // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion + int next_motion_time = object.motion_time - first_pixel_cycle + 4; + while(start < end) + { + int next_event_time = end; + + // is the next event a movement tick? + if(object.is_moving && next_motion_time < next_event_time) + { + next_event_time = next_motion_time; + } + + // is the next event a graphics trigger? + int next_copy = 160; + if(target.copy_flags) + { + if(object.position < 16 && target.copy_flags&1) + { + next_copy = 16; + } else if(object.position < 32 && target.copy_flags&2) + { + next_copy = 32; + } else if(object.position < 64 && target.copy_flags&4) + { + next_copy = 64; + } + } + + int next_copy_time = start + next_copy - object.position; + if(next_copy_time < next_event_time) next_event_time = next_copy_time; + + // the decision is to progress by length + const int length = next_event_time - start; + + target.draw_pixels(length); + + // the next interesting event is after next_event_time cycles, so progress + object.position = (object.position + length) % 160; + start = next_event_time; + + // if the event is a motion tick, apply + if(object.is_moving && start == next_motion_time) + { + perform_motion_step(object); + next_motion_time += 4; + } + + // if it's a draw trigger, trigger a draw + if(start == next_copy_time) + { + target.reset_pixels(); + } + } +} + #pragma mark - Player output void TIA::draw_player_visible(Player &player, Object &object, CollisionType collision_identity, int start, int end) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index fc59a98b0..9b183ee28 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -178,14 +178,7 @@ class TIA { // indicates whether this object is currently undergoing motion bool is_moving; - // receives a list of drawing events; 20 is a deliberately over-specified quantity - struct DrawingEvent { - int time; - int copy; - } drawing_events[20]; - int draw_event_read_pointer, draw_event_write_pointer; - - Object() : draw_event_read_pointer(0), draw_event_write_pointer(0), is_moving(false) {}; + Object() : is_moving(false) {}; } object_[5]; enum class MotionIndex : uint8_t { Ball, @@ -200,6 +193,9 @@ class TIA { inline void perform_motion_step(Object &object); // drawing methods and state + template void draw_object(T &, Object &, int start, int end); + template void draw_object_visible(T &, Object &, int start, int end); + inline void output_for_cycles(int number_of_cycles); inline void output_line(); From 7ab2358bba9f2d31deaf645ebd40db31a745ff6a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 20 Feb 2017 22:22:39 -0500 Subject: [PATCH 69/91] Made an attempt to reintroduce missiles. --- Machines/Atari2600/TIA.cpp | 224 ++++--------------------------------- Machines/Atari2600/TIA.hpp | 109 ++++++++++++++++-- 2 files changed, 117 insertions(+), 216 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 23cc6f698..926bdb1db 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -280,27 +280,28 @@ void TIA::set_playfield_ball_colour(uint8_t colour) void TIA::set_player_number_and_size(int player, uint8_t value) { + int size = 0; switch(value & 7) { case 0: case 1: case 2: case 3: case 4: - player_[player].size = 0; player_[player].copy_flags = value & 7; break; case 5: - player_[player].size = 1; + size = 1; player_[player].copy_flags = 0; break; case 6: - player_[player].size = 0; player_[player].copy_flags = 6; break; case 7: - player_[player].size = 2; + size = 2; player_[player].copy_flags = 0; break; } - missile_[player].size = (value >> 4)&3; + missile_[player].size = 1 << ((value >> 4)&3); + missile_[player].copy_flags = player_[player].copy_flags; + player_[player].adder = 4 >> size; } void TIA::set_player_graphic(int player, uint8_t value) @@ -342,18 +343,23 @@ void TIA::set_player_missile_colour(int player, uint8_t colour) void TIA::set_missile_enable(int missile, bool enabled) { + missile_[missile].enabled = enabled; } void TIA::set_missile_position(int missile) { + object_[(int)MotionIndex::Missile0 + missile].position = 0; } void TIA::set_missile_position_to_player(int missile) { + // TODO: implement this correctly; should be triggered by player counter hitting the appropriate point + object_[(int)MotionIndex::Missile0 + missile].position = object_[(int)MotionIndex::Player0 + missile].position + 5; } void TIA::set_missile_motion(int missile, uint8_t motion) { + object_[(int)MotionIndex::Missile0 + missile].motion = (motion >> 4)&0xf; } void TIA::set_ball_enable(bool enabled) @@ -371,7 +377,7 @@ void TIA::set_ball_position() object_[(int)MotionIndex::Ball].position = 0; // setting the ball position also triggers a draw - ball_.pixel_position = ball_.size; + ball_.reset_pixels(); } void TIA::set_ball_motion(uint8_t motion) @@ -643,7 +649,7 @@ int TIA::perform_border_motion(Object &object, int start, int end) return steps_taken; } -template void TIA::draw_object(T &target, Object &object, int start, int end) +template void TIA::draw_object(T &target, Object &object, const uint8_t collision_identity, int start, int end) { int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); @@ -661,7 +667,7 @@ template void TIA::draw_object(T &target, Object &object, int start, in // perform the visible part of the line, if any if(start < 224) { - draw_object_visible(target, object, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); + draw_object_visible(target, object, collision_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); } // move further if required @@ -672,7 +678,7 @@ template void TIA::draw_object(T &target, Object &object, int start, in } } -template void TIA::draw_object_visible(T &target, Object &object, int start, int end) +template void TIA::draw_object_visible(T &target, Object &object, const uint8_t collision_identity, int start, int end) { // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion int next_motion_time = object.motion_time - first_pixel_cycle + 4; @@ -708,7 +714,7 @@ template void TIA::draw_object_visible(T &target, Object &object, int s // the decision is to progress by length const int length = next_event_time - start; - target.draw_pixels(length); + target.draw_pixels(&collision_buffer_[start], length, collision_identity); // the next interesting event is after next_event_time cycles, so progress object.position = (object.position + length) % 160; @@ -731,209 +737,17 @@ template void TIA::draw_object_visible(T &target, Object &object, int s #pragma mark - Player output -void TIA::draw_player_visible(Player &player, Object &object, CollisionType collision_identity, int start, int end) -{ - int adder = 4 >> player.size; - - // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - int next_motion_time = object.motion_time - first_pixel_cycle + 4; - while(start < end) - { - int next_event_time = end; - - // is the next event a movement tick? - if(object.is_moving && next_motion_time < next_event_time) - { - next_event_time = next_motion_time; - } - - // is the next event a graphics trigger? - int next_copy = 160; - if(object.position < 16 && player.copy_flags&1) - { - next_copy = 16; - } else if(object.position < 32 && player.copy_flags&2) - { - next_copy = 32; - } else if(object.position < 64 && player.copy_flags&4) - { - next_copy = 64; - } - - int next_copy_time = start + next_copy - object.position; - if(next_copy_time < next_event_time) next_event_time = next_copy_time; - - // the decision is to progress by length - const int length = next_event_time - start; - - if(player.pixel_position < 32) - { - player.pixel_position &= ~(adder - 1); - if(player.graphic[player.graphic_index]) - { - int output_cursor = 0; - while(player.pixel_position < 32 && output_cursor < length) - { - int shift = (player.pixel_position >> 2) ^ player.reverse_mask; - collision_buffer_[start + output_cursor] |= ((player.graphic[player.graphic_index] >> shift)&1) * (uint8_t)collision_identity; - output_cursor++; - player.pixel_position += adder; - } - } - else - { - player.pixel_position = std::max(32, player.pixel_position + length * adder); - } - } - - // the next interesting event is after next_event_time cycles, so progress - object.position = (object.position + length) % 160; - start = next_event_time; - - // if the event is a motion tick, apply - if(object.is_moving && start == next_motion_time) - { - perform_motion_step(object); - next_motion_time += 4; - } - - // if it's a draw trigger, trigger a draw - if(start == next_copy_time) - { - player.pixel_position = 0; - } - } -} - void TIA::draw_player(Player &player, Object &object, CollisionType collision_identity, int start, int end) { - int adder = 4 >> player.size; - int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); - - // movement works across the entire screen, so do work that falls outside of the pixel area - if(start < first_pixel) - { - player.pixel_position = std::min(32, player.pixel_position + adder * perform_border_motion(object, start, std::max(end, first_pixel))); - } - - // don't continue to do any drawing if this window ends too early - if(end < first_pixel) return; - if(start < first_pixel) start = first_pixel; - if(start >= end) return; - - // perform the visible part of the line, if any - if(start < 224) - { - draw_player_visible(player, object, collision_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); - } - - // move further if required - if(object.is_moving && end >= 224 && object.motion_time < end) - { - perform_motion_step(object); - player.pixel_position = std::min(32, player.pixel_position + adder); - } + draw_object(player, object, (uint8_t)collision_identity, start, end); } -#pragma mark - Missile output - void TIA::draw_missile(Missile &missile, Object &object, CollisionType collision_identity, int start, int end) { + draw_object(missile, object, (uint8_t)collision_identity, start, end); } -void TIA::draw_missile_visible(Missile &missile, Object &object, CollisionType collision_identity, int start, int end) -{ -} - -#pragma mark - Ball output - void TIA::draw_ball(Object &object, int start, int end) { - int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); - - // movement works across the entire screen, so do work that falls outside of the pixel area - if(start < first_pixel) - { - ball_.pixel_position = std::max(0, ball_.pixel_position - perform_border_motion(object, start, std::max(end, first_pixel))); - } - - // don't continue to do any drawing if this window ends too early - if(end < first_pixel) return; - if(start < first_pixel) start = first_pixel; - if(start >= end) return; - - // perform the visible part of the line, if any - if(start < 224) - { - draw_ball_visible(object, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); - } - - // move further if required - if(object_[(int)MotionIndex::Ball].is_moving && end >= 224 && object_[(int)MotionIndex::Ball].motion_time < end) - { - perform_motion_step(object); - ball_.pixel_position = std::max(0, ball_.pixel_position - 1); - } -} - -void TIA::draw_ball_visible(Object &object, int start, int end) -{ - // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion - int next_motion_time = object.motion_time - first_pixel_cycle + 4; - while(start < end) - { - int next_event_time = end; - - // is the next event a movement tick? - if(object.is_moving && next_motion_time < next_event_time) - { - next_event_time = next_motion_time; - } - - // is the next event a graphics trigger? - if(ball_.enabled[ball_.enabled_index]) - { - int time_until_copy = 160 - object.position; - int next_copy_time = start + time_until_copy; - if(next_copy_time < next_event_time) next_event_time = next_copy_time; - } - - // the decision is to progress by length - const int length = next_event_time - start; - - if(ball_.pixel_position) - { - int output_cursor = 0; - if(ball_.enabled[ball_.enabled_index]) - { - while(ball_.pixel_position && output_cursor < length) - { - collision_buffer_[start + output_cursor] |= (uint8_t)CollisionType::Ball; - output_cursor++; - ball_.pixel_position--; - } - } - else - { - ball_.pixel_position = std::max(0, ball_.pixel_position - length); - } - } - - // the next interesting event is after next_event_time cycles, so progress - object.position = (object.position + length) % 160; - start = next_event_time; - - // if the event is a motion tick, apply - if(object.is_moving && start == next_motion_time) - { - perform_motion_step(object); - next_motion_time += 4; - } - - // if it's a draw trigger, trigger a draw - if(!object.position) - { - ball_.pixel_position = ball_.size; - } - } + draw_object(ball_, object, (uint8_t)CollisionType::Ball, start, end); } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 9b183ee28..10b731130 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -137,7 +137,7 @@ class TIA { // player state struct Player { - int size; // 0 = normal, 1 = double, 2 = quad + int adder; int copy_flags; // a bit field, corresponding to the first few values of NUSIZ uint8_t graphic[2]; // the player graphic; 1 = new, 0 = current int reverse_mask; // 7 for a reflected player, 0 for normal @@ -145,12 +145,75 @@ class TIA { int pixel_position; - Player() : size(0), copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0) {} + inline void skip_pixels(int count) + { + pixel_position = std::min(32, pixel_position + count * adder); + } + + inline void reset_pixels() + { + pixel_position = 0; + } + + inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + { + if(pixel_position == 32) return; + if(graphic[graphic_index]) + { + int output_cursor = 0; + while(pixel_position < 32 && output_cursor < count) + { + int shift = (pixel_position >> 2) ^ reverse_mask; + target[output_cursor] |= ((graphic[graphic_index] >> shift)&1) * collision_identity; + output_cursor++; + pixel_position += adder; + } + } + else + { + skip_pixels(count); + } + } + + Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0) {} } player_[2]; // missile state struct Missile { - int size; // 0 = 1 pixel, 1 = 2 pixels, etc + bool enabled; + int size; + int copy_flags; + + int pixel_position; + + inline void skip_pixels(int count) + { + pixel_position = std::max(0, pixel_position - count); + } + + inline void reset_pixels() + { + pixel_position = size; + } + + inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + { + if(!pixel_position) return; + if(enabled) + { + int output_cursor = 0; + while(pixel_position && output_cursor < count) + { + target[output_cursor] |= collision_identity; + output_cursor++; + pixel_position--; + } + } + else + { + skip_pixels(count); + } + } } missile_[2]; // ball state @@ -158,9 +221,39 @@ class TIA { bool enabled[2]; int enabled_index; int size; + const int copy_flags = 0; int pixel_position; + inline void skip_pixels(int count) + { + pixel_position = std::max(0, pixel_position - count); + } + + inline void reset_pixels() + { + pixel_position = size; + } + + inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + { + if(!pixel_position) return; + if(enabled[enabled_index]) + { + int output_cursor = 0; + while(pixel_position && output_cursor < count) + { + target[output_cursor] |= collision_identity; + output_cursor++; + pixel_position--; + } + } + else + { + skip_pixels(count); + } + } + Ball() : pixel_position(0), size(1), enabled_index(0) {} } ball_; @@ -193,22 +286,16 @@ class TIA { inline void perform_motion_step(Object &object); // drawing methods and state - template void draw_object(T &, Object &, int start, int end); - template void draw_object_visible(T &, Object &, int start, int end); + template void draw_object(T &, Object &, const uint8_t collision_identity, int start, int end); + template void draw_object_visible(T &, Object &, const uint8_t collision_identity, int start, int end); inline void output_for_cycles(int number_of_cycles); inline void output_line(); inline void draw_playfield(int start, int end); - inline void draw_player(Player &player, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_player_visible(Player &player, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_missile(Missile &missile, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_missile_visible(Missile &missile, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_ball(Object &object, int start, int end); - inline void draw_ball_visible(Object &object, int start, int end); int pixels_start_location_; uint8_t *pixel_target_; From 1bde0fed6f5e2a79e7a3697cece1f667b085e78e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 21 Feb 2017 07:37:20 -0500 Subject: [PATCH 70/91] Simplified relationship between Objects and the usage-specific components through inheritance. --- Machines/Atari2600/TIA.cpp | 73 ++++++++++++++++++++------------------ Machines/Atari2600/TIA.hpp | 57 +++++++++++++---------------- 2 files changed, 64 insertions(+), 66 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 926bdb1db..44131dc38 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -328,12 +328,12 @@ void TIA::set_player_position(int player) // one behind its real hardware value, creating the extra delay; and (ii) the player // code is written to start a draw upon wraparound from 159 to 0, so -1 is the // correct option rather than 159. - object_[(int)MotionIndex::Player0 + player].position = -1; + player_[player].position = -1; } void TIA::set_player_motion(int player, uint8_t motion) { - object_[(int)MotionIndex::Player0 + player].motion = (motion >> 4)&0xf; + player_[player].motion = (motion >> 4)&0xf; } void TIA::set_player_missile_colour(int player, uint8_t colour) @@ -348,18 +348,18 @@ void TIA::set_missile_enable(int missile, bool enabled) void TIA::set_missile_position(int missile) { - object_[(int)MotionIndex::Missile0 + missile].position = 0; + missile_[missile].position = 0; } void TIA::set_missile_position_to_player(int missile) { // TODO: implement this correctly; should be triggered by player counter hitting the appropriate point - object_[(int)MotionIndex::Missile0 + missile].position = object_[(int)MotionIndex::Player0 + missile].position + 5; + missile_[missile].position = player_[missile].position + 5; } void TIA::set_missile_motion(int missile, uint8_t motion) { - object_[(int)MotionIndex::Missile0 + missile].motion = (motion >> 4)&0xf; + missile_[missile].motion = (motion >> 4)&0xf; } void TIA::set_ball_enable(bool enabled) @@ -374,7 +374,7 @@ void TIA::set_ball_delay(bool delay) void TIA::set_ball_position() { - object_[(int)MotionIndex::Ball].position = 0; + ball_.position = 0; // setting the ball position also triggers a draw ball_.reset_pixels(); @@ -382,20 +382,20 @@ void TIA::set_ball_position() void TIA::set_ball_motion(uint8_t motion) { - object_[(int)MotionIndex::Ball].motion = (motion >> 4) & 0xf; + ball_.motion = (motion >> 4) & 0xf; } void TIA::move() { horizontal_blank_extend_ = true; - object_[0].is_moving = object_[1].is_moving = object_[2].is_moving = object_[3].is_moving = object_[4].is_moving = true; - object_[0].motion_step = object_[1].motion_step = object_[2].motion_step = object_[3].motion_step = object_[4].motion_step = 15; - object_[0].motion_time = object_[1].motion_time = object_[2].motion_time = object_[3].motion_time = object_[4].motion_time = (horizontal_counter_ + 3) & ~3; + player_[0].is_moving = player_[1].is_moving = missile_[0].is_moving = missile_[1].is_moving = ball_.is_moving = true; + player_[0].motion_step = player_[1].motion_step = missile_[0].motion_step = missile_[1].motion_step = ball_.motion_step = 15; + player_[0].motion_time = player_[1].motion_time = missile_[0].motion_time = missile_[1].motion_time = ball_.motion_time = (horizontal_counter_ + 3) & ~3; } void TIA::clear_motion() { - object_[0].motion = object_[1].motion = object_[2].motion = object_[3].motion = object_[4].motion = 0; + player_[0].motion = player_[1].motion = missile_[0].motion = missile_[1].motion = ball_.motion = 0; } uint8_t TIA::get_collision_flags(int offset) @@ -432,18 +432,23 @@ void TIA::output_for_cycles(int number_of_cycles) if(line_end_function_) line_end_function_(collision_buffer_.data()); memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_) horizontal_blank_extend_ = false; - for(int c = 0; c < 5; c++) object_[c].motion_time %= 228; + + ball_.motion_time %= 228; + player_[0].motion_time %= 228; + player_[1].motion_time %= 228; + missile_[0].motion_time %= 228; + missile_[1].motion_time %= 228; } // accumulate an OR'd version of the output into the collision buffer int latent_start = output_cursor + 4; int latent_end = horizontal_counter_ + 4; draw_playfield(latent_start, latent_end); - draw_player(player_[0], object_[(int)MotionIndex::Player0], CollisionType::Player0, output_cursor, horizontal_counter_); - draw_player(player_[1], object_[(int)MotionIndex::Player1], CollisionType::Player1, output_cursor, horizontal_counter_); - draw_missile(missile_[0], object_[(int)MotionIndex::Missile0], CollisionType::Missile0, output_cursor, horizontal_counter_); - draw_missile(missile_[1], object_[(int)MotionIndex::Missile1], CollisionType::Missile1, output_cursor, horizontal_counter_); - draw_ball(object_[(int)MotionIndex::Ball], output_cursor, horizontal_counter_); + draw_player(player_[0], CollisionType::Player0, output_cursor, horizontal_counter_); + draw_player(player_[1], CollisionType::Player1, output_cursor, horizontal_counter_); + draw_missile(missile_[0], CollisionType::Missile0, output_cursor, horizontal_counter_); + draw_missile(missile_[1], CollisionType::Missile1, output_cursor, horizontal_counter_); + draw_ball(output_cursor, horizontal_counter_); // convert to television signals @@ -649,14 +654,14 @@ int TIA::perform_border_motion(Object &object, int start, int end) return steps_taken; } -template void TIA::draw_object(T &target, Object &object, const uint8_t collision_identity, int start, int end) +template void TIA::draw_object(T &object, const uint8_t collision_identity, int start, int end) { int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); // movement works across the entire screen, so do work that falls outside of the pixel area if(start < first_pixel) { - target.skip_pixels(perform_border_motion(object, start, std::max(end, first_pixel))); + object.skip_pixels(perform_border_motion(object, start, std::max(end, first_pixel))); } // don't continue to do any drawing if this window ends too early @@ -667,18 +672,18 @@ template void TIA::draw_object(T &target, Object &object, const uint8_t // perform the visible part of the line, if any if(start < 224) { - draw_object_visible(target, object, collision_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); + draw_object_visible(object, collision_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); } // move further if required if(object.is_moving && end >= 224 && object.motion_time < end) { perform_motion_step(object); - target.skip_pixels(1); + object.skip_pixels(1); } } -template void TIA::draw_object_visible(T &target, Object &object, const uint8_t collision_identity, int start, int end) +template void TIA::draw_object_visible(T &object, const uint8_t collision_identity, int start, int end) { // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion int next_motion_time = object.motion_time - first_pixel_cycle + 4; @@ -694,15 +699,15 @@ template void TIA::draw_object_visible(T &target, Object &object, const // is the next event a graphics trigger? int next_copy = 160; - if(target.copy_flags) + if(object.copy_flags) { - if(object.position < 16 && target.copy_flags&1) + if(object.position < 16 && object.copy_flags&1) { next_copy = 16; - } else if(object.position < 32 && target.copy_flags&2) + } else if(object.position < 32 && object.copy_flags&2) { next_copy = 32; - } else if(object.position < 64 && target.copy_flags&4) + } else if(object.position < 64 && object.copy_flags&4) { next_copy = 64; } @@ -714,7 +719,7 @@ template void TIA::draw_object_visible(T &target, Object &object, const // the decision is to progress by length const int length = next_event_time - start; - target.draw_pixels(&collision_buffer_[start], length, collision_identity); + object.draw_pixels(&collision_buffer_[start], length, collision_identity); // the next interesting event is after next_event_time cycles, so progress object.position = (object.position + length) % 160; @@ -730,24 +735,24 @@ template void TIA::draw_object_visible(T &target, Object &object, const // if it's a draw trigger, trigger a draw if(start == next_copy_time) { - target.reset_pixels(); + object.reset_pixels(); } } } #pragma mark - Player output -void TIA::draw_player(Player &player, Object &object, CollisionType collision_identity, int start, int end) +void TIA::draw_player(Player &player, CollisionType collision_identity, int start, int end) { - draw_object(player, object, (uint8_t)collision_identity, start, end); + draw_object(player, (uint8_t)collision_identity, start, end); } -void TIA::draw_missile(Missile &missile, Object &object, CollisionType collision_identity, int start, int end) +void TIA::draw_missile(Missile &missile, CollisionType collision_identity, int start, int end) { - draw_object(missile, object, (uint8_t)collision_identity, start, end); + draw_object(missile, (uint8_t)collision_identity, start, end); } -void TIA::draw_ball(Object &object, int start, int end) +void TIA::draw_ball(int start, int end) { - draw_object(ball_, object, (uint8_t)CollisionType::Ball, start, end); + draw_object(ball_, (uint8_t)CollisionType::Ball, start, end); } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 10b731130..3b540147d 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -135,8 +135,24 @@ class TIA { // background_[1] on the right; otherwise background_[0] will be // output twice. + // objects + struct Object { + // the two programmer-set values + int position; + int motion; + + // motion_step_ is the current motion counter value; motion_time_ is the next time it will fire + int motion_step; + int motion_time; + + // indicates whether this object is currently undergoing motion + bool is_moving; + + Object() : is_moving(false) {}; + }; + // player state - struct Player { + struct Player: public Object { int adder; int copy_flags; // a bit field, corresponding to the first few values of NUSIZ uint8_t graphic[2]; // the player graphic; 1 = new, 0 = current @@ -179,7 +195,7 @@ class TIA { } player_[2]; // missile state - struct Missile { + struct Missile: public Object { bool enabled; int size; int copy_flags; @@ -217,7 +233,7 @@ class TIA { } missile_[2]; // ball state - struct Ball { + struct Ball: public Object { bool enabled[2]; int enabled_index; int size; @@ -257,45 +273,22 @@ class TIA { Ball() : pixel_position(0), size(1), enabled_index(0) {} } ball_; - // movement - bool horizontal_blank_extend_; - struct Object { - // the two programmer-set values - int position; - int motion; - - // motion_step_ is the current motion counter value; motion_time_ is the next time it will fire - int motion_step; - int motion_time; - - // indicates whether this object is currently undergoing motion - bool is_moving; - - Object() : is_moving(false) {}; - } object_[5]; - enum class MotionIndex : uint8_t { - Ball, - Player0, - Player1, - Missile0, - Missile1 - }; - // motion + bool horizontal_blank_extend_; inline int perform_border_motion(Object &object, int start, int end); inline void perform_motion_step(Object &object); // drawing methods and state - template void draw_object(T &, Object &, const uint8_t collision_identity, int start, int end); - template void draw_object_visible(T &, Object &, const uint8_t collision_identity, int start, int end); + template void draw_object(T &, const uint8_t collision_identity, int start, int end); + template void draw_object_visible(T &, const uint8_t collision_identity, int start, int end); inline void output_for_cycles(int number_of_cycles); inline void output_line(); inline void draw_playfield(int start, int end); - inline void draw_player(Player &player, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_missile(Missile &missile, Object &object, CollisionType collision_identity, int start, int end); - inline void draw_ball(Object &object, int start, int end); + inline void draw_player(Player &player, CollisionType collision_identity, int start, int end); + inline void draw_missile(Missile &missile, CollisionType collision_identity, int start, int end); + inline void draw_ball(int start, int end); int pixels_start_location_; uint8_t *pixel_target_; From d1dbf8c21f125574d931acd3484c119a74e7de25 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 21 Feb 2017 07:58:37 -0500 Subject: [PATCH 71/91] Missile to player lock is supposed to be a toggle; also factored out the commonalities of missile and ball drawing. --- Machines/Atari2600/Atari2600.cpp | 2 +- Machines/Atari2600/TIA.cpp | 16 ++++++-- Machines/Atari2600/TIA.hpp | 66 +++++++++++++++----------------- 3 files changed, 44 insertions(+), 40 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index d796fd983..a40f0ee55 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -175,7 +175,7 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin case 0x26: tia_->set_player_delay(decodedAddress - 0x25, (*value)&1); break; case 0x27: tia_->set_ball_delay((*value)&1); break; case 0x28: - case 0x29: update_video(); tia_->set_missile_position_to_player(decodedAddress - 0x28); break; + case 0x29: update_video(); tia_->set_missile_position_to_player(decodedAddress - 0x28, (*value)&2); break; case 0x2a: update_video(); tia_->move(); break; case 0x2b: update_video(); tia_->clear_motion(); break; case 0x2c: update_video(); tia_->clear_collision_flags(); break; diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 44131dc38..badbe4aeb 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -351,10 +351,20 @@ void TIA::set_missile_position(int missile) missile_[missile].position = 0; } -void TIA::set_missile_position_to_player(int missile) +void TIA::set_missile_position_to_player(int missile, bool lock) { - // TODO: implement this correctly; should be triggered by player counter hitting the appropriate point - missile_[missile].position = player_[missile].position + 5; + // TODO: implement this correctly; should be triggered by player counter hitting the appropriate point, and + // use additional storage position for enabled + if(lock) + { + missile_[missile].enabled = false; + missile_[missile].position = player_[missile].position + 5; +// printf("%d", missile); + } + else + { + missile_[missile].enabled = true; + } } void TIA::set_missile_motion(int missile, uint8_t motion) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 3b540147d..dcd3918b9 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -62,7 +62,7 @@ class TIA { void set_missile_enable(int missile, bool enabled); void set_missile_position(int missile); - void set_missile_position_to_player(int missile); + void set_missile_position_to_player(int missile, bool lock); void set_missile_motion(int missile, uint8_t motion); void set_ball_enable(bool enabled); @@ -194,13 +194,10 @@ class TIA { Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0) {} } player_[2]; - // missile state - struct Missile: public Object { - bool enabled; - int size; - int copy_flags; - + // common actor for things that appear as a horizontal run of pixels + struct HorizontalRun: public Object { int pixel_position; + int size; inline void skip_pixels(int count) { @@ -212,57 +209,54 @@ class TIA { pixel_position = size; } + inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + { + int output_cursor = 0; + while(pixel_position && output_cursor < count) + { + target[output_cursor] |= collision_identity; + output_cursor++; + pixel_position--; + } + } + + HorizontalRun() : pixel_position(0), size(1) {} + }; + + + // missile state + struct Missile: public HorizontalRun { + bool enabled; + int copy_flags; + inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(!pixel_position) return; if(enabled) { - int output_cursor = 0; - while(pixel_position && output_cursor < count) - { - target[output_cursor] |= collision_identity; - output_cursor++; - pixel_position--; - } + HorizontalRun::draw_pixels(target, count, collision_identity); } else { skip_pixels(count); } } + + Missile() : enabled(false), copy_flags(0) {} } missile_[2]; // ball state - struct Ball: public Object { + struct Ball: public HorizontalRun { bool enabled[2]; int enabled_index; - int size; const int copy_flags = 0; - int pixel_position; - - inline void skip_pixels(int count) - { - pixel_position = std::max(0, pixel_position - count); - } - - inline void reset_pixels() - { - pixel_position = size; - } - inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(!pixel_position) return; if(enabled[enabled_index]) { - int output_cursor = 0; - while(pixel_position && output_cursor < count) - { - target[output_cursor] |= collision_identity; - output_cursor++; - pixel_position--; - } + HorizontalRun::draw_pixels(target, count, collision_identity); } else { @@ -270,7 +264,7 @@ class TIA { } } - Ball() : pixel_position(0), size(1), enabled_index(0) {} + Ball() : enabled_index(0), enabled{false, false} {} } ball_; // motion From 36396b3d62d8b1e46e84c46d38902c5a8aadb95a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 21 Feb 2017 20:45:20 -0500 Subject: [PATCH 72/91] Made a slightly better, albeit still inaccurate, version of missile-player lock.Enough for Combat to do reasonable things. --- Machines/Atari2600/TIA.cpp | 12 ++---------- Machines/Atari2600/TIA.hpp | 3 ++- 2 files changed, 4 insertions(+), 11 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index badbe4aeb..34944e497 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -355,16 +355,8 @@ void TIA::set_missile_position_to_player(int missile, bool lock) { // TODO: implement this correctly; should be triggered by player counter hitting the appropriate point, and // use additional storage position for enabled - if(lock) - { - missile_[missile].enabled = false; - missile_[missile].position = player_[missile].position + 5; -// printf("%d", missile); - } - else - { - missile_[missile].enabled = true; - } + if(missile_[missile].locked_to_player && !lock) missile_[missile].position = player_[missile].position + 1 + 16/player_[missile].adder; + missile_[missile].locked_to_player = lock; } void TIA::set_missile_motion(int missile, uint8_t motion) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index dcd3918b9..7ae5d428d 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -227,12 +227,13 @@ class TIA { // missile state struct Missile: public HorizontalRun { bool enabled; + bool locked_to_player; int copy_flags; inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(!pixel_position) return; - if(enabled) + if(enabled && !locked_to_player) { HorizontalRun::draw_pixels(target, count, collision_identity); } From f4447fd9cd7179f32153152400ce23cca79475ef Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 21 Feb 2017 21:53:09 -0500 Subject: [PATCH 73/91] Attempted to fix failure of sprites properly to wrap when performing motion. --- Machines/Atari2600/TIA.cpp | 30 ++++++++++++------------------ Machines/Atari2600/TIA.hpp | 4 ++-- 2 files changed, 14 insertions(+), 20 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 34944e497..3b373c2a3 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -629,31 +629,27 @@ void TIA::draw_playfield(int start, int end) #pragma mark - Motion -void TIA::perform_motion_step(Object &object) +template void TIA::perform_motion_step(T &object) { if((object.motion_step ^ (object.motion ^ 8)) == 0xf) object.is_moving = false; else { - object.position ++; + if(object.position == 159) object.reset_pixels(); + else if(object.position == 15 && object.copy_flags&1) object.reset_pixels(); + else if(object.position == 31 && object.copy_flags&2) object.reset_pixels(); + else if(object.position == 63 && object.copy_flags&4) object.reset_pixels(); + else object.skip_pixels(1); + object.position = (object.position + 1) % 160; object.motion_step --; object.motion_time += 4; } } -int TIA::perform_border_motion(Object &object, int start, int end) +template void TIA::perform_border_motion(T &object, int start, int end) { - if(!object.is_moving) return 0; - - int steps_taken = 0; while(object.is_moving && object.motion_time < end) - { - perform_motion_step(object); - steps_taken++; - } - object.position %= 160; - - return steps_taken; + perform_motion_step(object); } template void TIA::draw_object(T &object, const uint8_t collision_identity, int start, int end) @@ -663,7 +659,7 @@ template void TIA::draw_object(T &object, const uint8_t collision_ident // movement works across the entire screen, so do work that falls outside of the pixel area if(start < first_pixel) { - object.skip_pixels(perform_border_motion(object, start, std::max(end, first_pixel))); + perform_border_motion(object, start, std::max(end, first_pixel)); } // don't continue to do any drawing if this window ends too early @@ -680,8 +676,7 @@ template void TIA::draw_object(T &object, const uint8_t collision_ident // move further if required if(object.is_moving && end >= 224 && object.motion_time < end) { - perform_motion_step(object); - object.skip_pixels(1); + perform_motion_step(object); } } @@ -733,9 +728,8 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi perform_motion_step(object); next_motion_time += 4; } - // if it's a draw trigger, trigger a draw - if(start == next_copy_time) + else if(start == next_copy_time) { object.reset_pixels(); } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 7ae5d428d..543f40aff 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -270,8 +270,8 @@ class TIA { // motion bool horizontal_blank_extend_; - inline int perform_border_motion(Object &object, int start, int end); - inline void perform_motion_step(Object &object); + template void perform_border_motion(T &object, int start, int end); + template void perform_motion_step(T &object); // drawing methods and state template void draw_object(T &, const uint8_t collision_identity, int start, int end); From 7019d396d0ce907f66895cf850dd85817c55e5ee Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 21 Feb 2017 22:04:27 -0500 Subject: [PATCH 74/91] Threw in some asserts, discovering a bug in missile positioning. --- Machines/Atari2600/Atari2600.cpp | 2 +- Machines/Atari2600/TIA.cpp | 12 ++++++++++++ 2 files changed, 13 insertions(+), 1 deletion(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index a40f0ee55..67b99d18d 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -159,7 +159,7 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin case 0x10: case 0x11: update_video(); tia_->set_player_position(decodedAddress - 0x10); break; case 0x12: - case 0x13: update_video(); tia_->set_missile_position(decodedAddress - 0x13); break; + case 0x13: update_video(); tia_->set_missile_position(decodedAddress - 0x12); break; case 0x14: update_video(); tia_->set_ball_position(); break; case 0x1b: case 0x1c: update_video(); tia_->set_player_graphic(decodedAddress - 0x1b, *value); break; diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 3b373c2a3..da7f8b5ce 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -7,6 +7,7 @@ // #include "TIA.hpp" +#include using namespace Atari2600; namespace { @@ -280,6 +281,7 @@ void TIA::set_playfield_ball_colour(uint8_t colour) void TIA::set_player_number_and_size(int player, uint8_t value) { + assert(player >= 0 && player < 2); int size = 0; switch(value & 7) { @@ -306,6 +308,7 @@ void TIA::set_player_number_and_size(int player, uint8_t value) void TIA::set_player_graphic(int player, uint8_t value) { + assert(player >= 0 && player < 2); player_[player].graphic[1] = value; player_[player^1].graphic[0] = player_[player^1].graphic[1]; if(player) ball_.enabled[0] = ball_.enabled[1]; @@ -313,16 +316,19 @@ void TIA::set_player_graphic(int player, uint8_t value) void TIA::set_player_reflected(int player, bool reflected) { + assert(player >= 0 && player < 2); player_[player].reverse_mask = reflected ? 7 : 0; } void TIA::set_player_delay(int player, bool delay) { + assert(player >= 0 && player < 2); player_[player].graphic_index = delay ? 0 : 1; } void TIA::set_player_position(int player) { + assert(player >= 0 && player < 2); // players have an extra clock of delay before output and don't display upon reset; // both aims are achieved by setting to -1 because: (i) it causes the clock to be // one behind its real hardware value, creating the extra delay; and (ii) the player @@ -333,26 +339,31 @@ void TIA::set_player_position(int player) void TIA::set_player_motion(int player, uint8_t motion) { + assert(player >= 0 && player < 2); player_[player].motion = (motion >> 4)&0xf; } void TIA::set_player_missile_colour(int player, uint8_t colour) { + assert(player >= 0 && player < 2); colour_palette_[(int)ColourIndex::PlayerMissile0 + player] = colour; } void TIA::set_missile_enable(int missile, bool enabled) { + assert(missile >= 0 && missile < 2); missile_[missile].enabled = enabled; } void TIA::set_missile_position(int missile) { + assert(missile >= 0 && missile < 2); missile_[missile].position = 0; } void TIA::set_missile_position_to_player(int missile, bool lock) { + assert(missile >= 0 && missile < 2); // TODO: implement this correctly; should be triggered by player counter hitting the appropriate point, and // use additional storage position for enabled if(missile_[missile].locked_to_player && !lock) missile_[missile].position = player_[missile].position + 1 + 16/player_[missile].adder; @@ -361,6 +372,7 @@ void TIA::set_missile_position_to_player(int missile, bool lock) void TIA::set_missile_motion(int missile, uint8_t motion) { + assert(missile >= 0 && missile < 2); missile_[missile].motion = (motion >> 4)&0xf; } From b769f22ca0f195d086ecee7e550b35a1df9d29c1 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 21 Feb 2017 22:26:20 -0500 Subject: [PATCH 75/91] Switched back to the collision_buffer_ being part of the TIA object, added one more assert. --- Machines/Atari2600/TIA.cpp | 7 +++---- Machines/Atari2600/TIA.hpp | 3 +-- 2 files changed, 4 insertions(+), 6 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index da7f8b5ce..63b3bb569 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -128,8 +128,6 @@ TIA::TIA(bool create_crt) : colour_mask_by_mode_collision_flags_[(int)ColourMode::OnTop][c] = (uint8_t)ColourIndex::PlayfieldBall; } } - - collision_buffer_.resize(160); } TIA::TIA() : TIA(true) {} @@ -237,6 +235,7 @@ void TIA::set_background_colour(uint8_t colour) void TIA::set_playfield(uint16_t offset, uint8_t value) { + assert(offset >= 0 && offset < 3); switch(offset) { case 0: @@ -443,8 +442,8 @@ void TIA::output_for_cycles(int number_of_cycles) if(!output_cursor) { - if(line_end_function_) line_end_function_(collision_buffer_.data()); - memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_) + if(line_end_function_) line_end_function_(collision_buffer_); + memset(collision_buffer_, 0, sizeof(collision_buffer_)); horizontal_blank_extend_ = false; ball_.motion_time %= 228; diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 543f40aff..1ed9040ab 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -89,8 +89,7 @@ class TIA { int output_mode_; // keeps track of the target pixel buffer for this line and when it was acquired, and a corresponding collision buffer -// alignas(alignof(uint32_t)) uint8_t collision_buffer_[160]; - std::vector collision_buffer_; + alignas(alignof(uint32_t)) uint8_t collision_buffer_[160]; enum class CollisionType : uint8_t { Playfield = (1 << 0), Ball = (1 << 1), From 144d6b70d925749877e53e504e85a12caa863178 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 07:14:30 -0500 Subject: [PATCH 76/91] Minor cleaning. --- Machines/Atari2600/TIA.cpp | 36 +++++------------------------------- Machines/Atari2600/TIA.hpp | 10 ++-------- 2 files changed, 7 insertions(+), 39 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 63b3bb569..177ddc830 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -176,15 +176,6 @@ void TIA::set_output_mode(Atari2600::TIA::OutputMode output_mode) /* speaker_->set_input_rate((float)(get_clock_rate() / 38.0));*/ } -TIA::~TIA() -{ -} - - // justification for +5: "we need to wait at least 71 [clocks] before the HMOVE operation is complete"; - // which will take 16*4 + 2 = 66 cycles from the first compare, implying the first compare must be - // in five cycles from now - - void TIA::run_for_cycles(int number_of_cycles) { // if part way through a line, definitely perform a partial, at most up to the end of the line @@ -457,11 +448,11 @@ void TIA::output_for_cycles(int number_of_cycles) int latent_start = output_cursor + 4; int latent_end = horizontal_counter_ + 4; draw_playfield(latent_start, latent_end); - draw_player(player_[0], CollisionType::Player0, output_cursor, horizontal_counter_); - draw_player(player_[1], CollisionType::Player1, output_cursor, horizontal_counter_); - draw_missile(missile_[0], CollisionType::Missile0, output_cursor, horizontal_counter_); - draw_missile(missile_[1], CollisionType::Missile1, output_cursor, horizontal_counter_); - draw_ball(output_cursor, horizontal_counter_); + draw_object(player_[0], (uint8_t)CollisionType::Player0, output_cursor, horizontal_counter_); + draw_object(player_[1], (uint8_t)CollisionType::Player1, output_cursor, horizontal_counter_); + draw_object(missile_[0], (uint8_t)CollisionType::Missile0, output_cursor, horizontal_counter_); + draw_object(missile_[1], (uint8_t)CollisionType::Missile1, output_cursor, horizontal_counter_); + draw_object(ball_, (uint8_t)CollisionType::Ball, output_cursor, horizontal_counter_); // convert to television signals @@ -746,20 +737,3 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi } } } - -#pragma mark - Player output - -void TIA::draw_player(Player &player, CollisionType collision_identity, int start, int end) -{ - draw_object(player, (uint8_t)collision_identity, start, end); -} - -void TIA::draw_missile(Missile &missile, CollisionType collision_identity, int start, int end) -{ - draw_object(missile, (uint8_t)collision_identity, start, end); -} - -void TIA::draw_ball(int start, int end) -{ - draw_object(ball_, (uint8_t)CollisionType::Ball, start, end); -} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 1ed9040ab..3914092c0 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -17,13 +17,10 @@ namespace Atari2600 { class TIA { public: TIA(); - ~TIA(); - // The supplied hook is for unit testing only; if instantiated with a line_end_function then it will // be called with the latest collision buffer upon the conclusion of each line. What's a collision // buffer? It's an implementation detail. If you're not writing a unit test, leave it alone. TIA(std::function line_end_function); - TIA(bool create_crt); enum class OutputMode { NTSC, PAL @@ -79,6 +76,7 @@ class TIA { virtual std::shared_ptr get_crt() { return crt_; } private: + TIA(bool create_crt); std::shared_ptr crt_; std::function line_end_function_; @@ -275,15 +273,11 @@ class TIA { // drawing methods and state template void draw_object(T &, const uint8_t collision_identity, int start, int end); template void draw_object_visible(T &, const uint8_t collision_identity, int start, int end); + inline void draw_playfield(int start, int end); inline void output_for_cycles(int number_of_cycles); inline void output_line(); - inline void draw_playfield(int start, int end); - inline void draw_player(Player &player, CollisionType collision_identity, int start, int end); - inline void draw_missile(Missile &missile, CollisionType collision_identity, int start, int end); - inline void draw_ball(int start, int end); - int pixels_start_location_; uint8_t *pixel_target_; inline void output_pixels(int start, int end); From 21abf4e9fc1007af7a81c9f794219deda2489664 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 07:29:48 -0500 Subject: [PATCH 77/91] Enshrined a terminology switch, albeit without any flow change behind it. --- Machines/Atari2600/TIA.cpp | 6 +++++- Machines/Atari2600/TIA.hpp | 16 ++++++++-------- 2 files changed, 13 insertions(+), 9 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 177ddc830..6ec63b8db 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -718,7 +718,11 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi // the decision is to progress by length const int length = next_event_time - start; - object.draw_pixels(&collision_buffer_[start], length, collision_identity); + // TODO: the problem with this is that it uses the enabled/pixel state of each object four cycles early; + // an appropriate solution would probably be to capture the drawing request into a queue and honour them outside + // this loop, clipped to the real output parameters. Assuming all state consumed by draw_pixels is captured, + // and mutated now then also queueing resets and skips shouldn't be necessary. + object.enqueue_pixels(&collision_buffer_[start], length, collision_identity); // the next interesting event is after next_event_time cycles, so progress object.position = (object.position + length) % 160; diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 3914092c0..45e051133 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -158,7 +158,7 @@ class TIA { int pixel_position; - inline void skip_pixels(int count) + inline void skip_pixels(const int count) { pixel_position = std::min(32, pixel_position + count * adder); } @@ -168,7 +168,7 @@ class TIA { pixel_position = 0; } - inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(pixel_position == 32) return; if(graphic[graphic_index]) @@ -196,7 +196,7 @@ class TIA { int pixel_position; int size; - inline void skip_pixels(int count) + inline void skip_pixels(const int count) { pixel_position = std::max(0, pixel_position - count); } @@ -206,7 +206,7 @@ class TIA { pixel_position = size; } - inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { int output_cursor = 0; while(pixel_position && output_cursor < count) @@ -227,12 +227,12 @@ class TIA { bool locked_to_player; int copy_flags; - inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(!pixel_position) return; if(enabled && !locked_to_player) { - HorizontalRun::draw_pixels(target, count, collision_identity); + HorizontalRun::enqueue_pixels(target, count, collision_identity); } else { @@ -249,12 +249,12 @@ class TIA { int enabled_index; const int copy_flags = 0; - inline void draw_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(!pixel_position) return; if(enabled[enabled_index]) { - HorizontalRun::draw_pixels(target, count, collision_identity); + HorizontalRun::enqueue_pixels(target, count, collision_identity); } else { From 4f5f191cd693918c0ab4a032677d5fdad864c505 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 07:33:36 -0500 Subject: [PATCH 78/91] Fixed: will no longer attempt to output pixels from before the pixel part of a line on which sync was disabled abruptly. --- Machines/Atari2600/TIA.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 6ec63b8db..fb582ebf5 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -534,6 +534,7 @@ void TIA::output_for_cycles(int number_of_cycles) void TIA::output_pixels(int start, int end) { + start = std::min(start, pixels_start_location_); int target_position = start - pixels_start_location_; if(start < first_pixel_cycle+8 && horizontal_blank_extend_) From 1d03793f22b63c0555015db4a16d66919305425e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 07:35:09 -0500 Subject: [PATCH 79/91] Fixed potential race condition: ensure the queue is disposed of synchronously because otherwise there'll be a potential dangling reference to self. --- OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m b/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m index efc3719b0..82373c824 100644 --- a/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m +++ b/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m @@ -104,7 +104,7 @@ static void audioOutputCallback( - (void)dealloc { - if(_audioQueue) AudioQueueDispose(_audioQueue, NO); + if(_audioQueue) AudioQueueDispose(_audioQueue, YES); } #pragma mark - Audio enqueuer From 6120dae61a177727dc363aafb3ca7a0840d78230 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 07:39:11 -0500 Subject: [PATCH 80/91] While I'm using the hacky approach to player/missile synchronisation, I need to seed adder. --- Machines/Atari2600/TIA.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 45e051133..da6a77d86 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -188,7 +188,7 @@ class TIA { } } - Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0) {} + Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0), adder(4) {} } player_[2]; // common actor for things that appear as a horizontal run of pixels From 4c947ad5538e854fdeb2d71853d68fe0aa6beb71 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 21:12:59 -0500 Subject: [PATCH 81/91] Attempted to resolve risk of an audio callback being in progress when -dealloc is received. --- .../Mac/Clock Signal/Audio/CSAudioQueue.m | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m b/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m index 82373c824..1c6bd4990 100644 --- a/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m +++ b/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m @@ -12,6 +12,8 @@ #define AudioQueueBufferMaxLength 8192 #define NumberOfStoredAudioQueueBuffer 16 +static NSLock *CSAudioQueueDeallocLock; + @implementation CSAudioQueue { AudioQueueRef _audioQueue; @@ -45,7 +47,11 @@ static void audioOutputCallback( AudioQueueRef inAQ, AudioQueueBufferRef inBuffer) { - [(__bridge CSAudioQueue *)inUserData audioQueue:inAQ didCallbackWithBuffer:inBuffer]; + if([CSAudioQueueDeallocLock tryLock]) + { + [(__bridge CSAudioQueue *)inUserData audioQueue:inAQ didCallbackWithBuffer:inBuffer]; + [CSAudioQueueDeallocLock unlock]; + } } #pragma mark - Standard object lifecycle @@ -56,6 +62,11 @@ static void audioOutputCallback( if(self) { + if(!CSAudioQueueDeallocLock) + { + CSAudioQueueDeallocLock = [[NSLock alloc] init]; + } + _samplingRate = samplingRate; // determine preferred buffer sizes @@ -104,7 +115,9 @@ static void audioOutputCallback( - (void)dealloc { - if(_audioQueue) AudioQueueDispose(_audioQueue, YES); + [CSAudioQueueDeallocLock lock]; + if(_audioQueue) AudioQueueDispose(_audioQueue, true); + [CSAudioQueueDeallocLock unlock]; } #pragma mark - Audio enqueuer From 8291a63d5f52275067b594d536cb48c0cdcca93e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 21:15:37 -0500 Subject: [PATCH 82/91] Fixed loss of audio when switching to PAL. --- Machines/Atari2600/Atari2600.cpp | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index 67b99d18d..5e10d023b 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -319,17 +319,19 @@ void Machine::crt_did_end_batch_of_frames(Outputs::CRT::CRT *crt, unsigned int n } is_ntsc_ ^= true; + double clock_rate; if(is_ntsc_) { - set_clock_rate(NTSC_clock_rate); + clock_rate = NTSC_clock_rate; tia_->set_output_mode(TIA::OutputMode::NTSC); } else { - set_clock_rate(PAL_clock_rate); + clock_rate = PAL_clock_rate; tia_->set_output_mode(TIA::OutputMode::PAL); } - speaker_->set_input_rate((float)(get_clock_rate() / 38.0)); + speaker_->set_input_rate((float)(clock_rate / 38.0)); + set_clock_rate(clock_rate); } } From 2f0c923c29a3885733d170f67d08fcd0f9ba1ebe Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 21:42:10 -0500 Subject: [PATCH 83/91] Switched away from @synchronized as it appears possibly to be the lock used during -dealloc, creating deadlock with the CSAudioQueueDeallocLock. --- .../Mac/Clock Signal/Audio/CSAudioQueue.m | 53 ++++++++++--------- 1 file changed, 27 insertions(+), 26 deletions(-) diff --git a/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m b/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m index 1c6bd4990..83280ad8b 100644 --- a/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m +++ b/OSBindings/Mac/Clock Signal/Audio/CSAudioQueue.m @@ -17,8 +17,8 @@ static NSLock *CSAudioQueueDeallocLock; @implementation CSAudioQueue { AudioQueueRef _audioQueue; - AudioQueueBufferRef _storedBuffers[NumberOfStoredAudioQueueBuffer]; + NSLock *_storedBuffersLock; } #pragma mark - AudioQueue callbacks @@ -27,18 +27,18 @@ static NSLock *CSAudioQueueDeallocLock; { [self.delegate audioQueueIsRunningDry:self]; - @synchronized(self) + [_storedBuffersLock lock]; + for(int c = 0; c < NumberOfStoredAudioQueueBuffer; c++) { - for(int c = 0; c < NumberOfStoredAudioQueueBuffer; c++) + if(!_storedBuffers[c] || buffer->mAudioDataBytesCapacity > _storedBuffers[c]->mAudioDataBytesCapacity) { - if(!_storedBuffers[c] || buffer->mAudioDataBytesCapacity > _storedBuffers[c]->mAudioDataBytesCapacity) - { - if(_storedBuffers[c]) AudioQueueFreeBuffer(_audioQueue, _storedBuffers[c]); - _storedBuffers[c] = buffer; - return; - } + if(_storedBuffers[c]) AudioQueueFreeBuffer(_audioQueue, _storedBuffers[c]); + _storedBuffers[c] = buffer; + [_storedBuffersLock unlock]; + return; } } + [_storedBuffersLock unlock]; AudioQueueFreeBuffer(_audioQueue, buffer); } @@ -66,6 +66,7 @@ static void audioOutputCallback( { CSAudioQueueDeallocLock = [[NSLock alloc] init]; } + _storedBuffersLock = [[NSLock alloc] init]; _samplingRate = samplingRate; @@ -126,28 +127,28 @@ static void audioOutputCallback( { size_t bufferBytes = lengthInSamples * sizeof(int16_t); - @synchronized(self) + [_storedBuffersLock lock]; + for(int c = 0; c < NumberOfStoredAudioQueueBuffer; c++) { - for(int c = 0; c < NumberOfStoredAudioQueueBuffer; c++) + if(_storedBuffers[c] && _storedBuffers[c]->mAudioDataBytesCapacity >= bufferBytes) { - if(_storedBuffers[c] && _storedBuffers[c]->mAudioDataBytesCapacity >= bufferBytes) - { - memcpy(_storedBuffers[c]->mAudioData, buffer, bufferBytes); - _storedBuffers[c]->mAudioDataByteSize = (UInt32)bufferBytes; + memcpy(_storedBuffers[c]->mAudioData, buffer, bufferBytes); + _storedBuffers[c]->mAudioDataByteSize = (UInt32)bufferBytes; - AudioQueueEnqueueBuffer(_audioQueue, _storedBuffers[c], 0, NULL); - _storedBuffers[c] = NULL; - return; - } + AudioQueueEnqueueBuffer(_audioQueue, _storedBuffers[c], 0, NULL); + _storedBuffers[c] = NULL; + [_storedBuffersLock unlock]; + return; } - - AudioQueueBufferRef newBuffer; - AudioQueueAllocateBuffer(_audioQueue, (UInt32)bufferBytes * 2, &newBuffer); - memcpy(newBuffer->mAudioData, buffer, bufferBytes); - newBuffer->mAudioDataByteSize = (UInt32)bufferBytes; - - AudioQueueEnqueueBuffer(_audioQueue, newBuffer, 0, NULL); } + [_storedBuffersLock unlock]; + + AudioQueueBufferRef newBuffer; + AudioQueueAllocateBuffer(_audioQueue, (UInt32)bufferBytes * 2, &newBuffer); + memcpy(newBuffer->mAudioData, buffer, bufferBytes); + newBuffer->mAudioDataByteSize = (UInt32)bufferBytes; + + AudioQueueEnqueueBuffer(_audioQueue, newBuffer, 0, NULL); } #pragma mark - Sampling Rate getters From 82f392fada274cf25502b8bc68715368320c2833 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 22 Feb 2017 21:54:49 -0500 Subject: [PATCH 84/91] This should be the other way around. I want whichever is later. --- Machines/Atari2600/TIA.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index fb582ebf5..356e48116 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -534,7 +534,7 @@ void TIA::output_for_cycles(int number_of_cycles) void TIA::output_pixels(int start, int end) { - start = std::min(start, pixels_start_location_); + start = std::max(start, pixels_start_location_); int target_position = start - pixels_start_location_; if(start < first_pixel_cycle+8 && horizontal_blank_extend_) From 77ed4ddc0528d2758de0c8e60c980d74a424d76e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 23 Feb 2017 21:08:32 -0500 Subject: [PATCH 85/91] Slightly simplified ready line release logic. --- Machines/Atari2600/Atari2600.cpp | 15 +++++---------- 1 file changed, 5 insertions(+), 10 deletions(-) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index 5e10d023b..25a8b9792 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -63,9 +63,6 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin cycles_since_speaker_update_ += cycles_run_for; cycles_since_video_update_ += cycles_run_for; - if(!tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_)) - set_ready_line(false); - if(operation != CPU6502::BusOperation::Ready) { // check for a paging access @@ -134,14 +131,11 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin } else { const uint16_t decodedAddress = address & 0x3f; switch(decodedAddress) { - case 0x00: update_video(); tia_->set_sync(*value & 0x02); break; - case 0x01: update_video(); tia_->set_blank(*value & 0x02); break; + case 0x00: update_video(); tia_->set_sync(*value & 0x02); break; + case 0x01: update_video(); tia_->set_blank(*value & 0x02); break; - case 0x02: - if(tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_)) - set_ready_line(true); - break; - case 0x03: update_video(); tia_->reset_horizontal_counter(); break; + case 0x02: set_ready_line(true); break; + case 0x03: update_video(); tia_->reset_horizontal_counter(); break; // TODO: audio will now be out of synchronisation — fix case 0x04: @@ -204,6 +198,7 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin } } + if(!tia_->get_cycles_until_horizontal_blank(cycles_since_video_update_)) set_ready_line(false); mos6532_.run_for_cycles(cycles_run_for / 3); return cycles_run_for / 3; From 8c9062857c83c1e602e9cbeb2c3f1fb6b21cd9db Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 25 Feb 2017 17:10:24 -0500 Subject: [PATCH 86/91] Added a single-slot queue for player objects to defer drawing, thereby deferring pixel lookup. Which I think is correct. Though more slots might be needed. --- Machines/Atari2600/TIA.cpp | 23 +++++++++-- Machines/Atari2600/TIA.hpp | 83 +++++++++++++++++++++++++++++--------- 2 files changed, 85 insertions(+), 21 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 356e48116..0fa51316d 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -659,6 +659,8 @@ template void TIA::draw_object(T &object, const uint8_t collision_ident { int first_pixel = first_pixel_cycle - 4 + (horizontal_blank_extend_ ? 8 : 0); + object.dequeue_pixels(collision_buffer_, collision_identity, end - first_pixel_cycle); + // movement works across the entire screen, so do work that falls outside of the pixel area if(start < first_pixel) { @@ -673,7 +675,7 @@ template void TIA::draw_object(T &object, const uint8_t collision_ident // perform the visible part of the line, if any if(start < 224) { - draw_object_visible(object, collision_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160)); + draw_object_visible(object, collision_identity, start - first_pixel_cycle + 4, std::min(end - first_pixel_cycle + 4, 160), end - first_pixel_cycle); } // move further if required @@ -683,7 +685,7 @@ template void TIA::draw_object(T &object, const uint8_t collision_ident } } -template void TIA::draw_object_visible(T &object, const uint8_t collision_identity, int start, int end) +template void TIA::draw_object_visible(T &object, const uint8_t collision_identity, int start, int end, int time_now) { // perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion int next_motion_time = object.motion_time - first_pixel_cycle + 4; @@ -723,7 +725,22 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi // an appropriate solution would probably be to capture the drawing request into a queue and honour them outside // this loop, clipped to the real output parameters. Assuming all state consumed by draw_pixels is captured, // and mutated now then also queueing resets and skips shouldn't be necessary. - object.enqueue_pixels(&collision_buffer_[start], length, collision_identity); + if(next_event_time > time_now) + { + if(start < time_now) + { + object.output_pixels(&collision_buffer_[start], time_now - start, collision_identity); + object.enqueue_pixels(time_now, next_event_time); + } + else + { + object.enqueue_pixels(start, next_event_time); + } + } + else + { + object.output_pixels(&collision_buffer_[start], length, collision_identity); + } // the next interesting event is after next_event_time cycles, so progress object.position = (object.position + length) % 160; diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index da6a77d86..4114a44be 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -133,7 +133,7 @@ class TIA { // output twice. // objects - struct Object { + template struct Object { // the two programmer-set values int position; int motion; @@ -146,10 +146,31 @@ class TIA { bool is_moving; Object() : is_moving(false) {}; + + void dequeue_pixels(uint8_t *const target, const uint8_t collision_identity, const int time_now) + { +// if(enqueued_start_ != enqueued_end_) +// { +// static_cast(this)->output_pixels(&target[enqueued_start_], enqueued_end_ - enqueued_start_, collision_identity); +// enqueued_end_ = enqueued_start_ = 0; +// } + } + + void enqueue_pixels(const int start, const int end) + { +// enqueued_start_ = start; +// enqueued_end_ = end; + static_cast(this)->skip_pixels(end - start); + } + + private: +// int enqueued_start_, enqueued_end_; }; // player state - struct Player: public Object { + struct Player: public Object { + Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0), adder(4) {} + int adder; int copy_flags; // a bit field, corresponding to the first few values of NUSIZ uint8_t graphic[2]; // the player graphic; 1 = new, 0 = current @@ -168,11 +189,43 @@ class TIA { pixel_position = 0; } - inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { - if(pixel_position == 32) return; - if(graphic[graphic_index]) + output_pixels(target, count, collision_identity, pixel_position, adder, reverse_mask); + skip_pixels(count); + } + + void dequeue_pixels(uint8_t *const target, const uint8_t collision_identity, const int time_now) + { + if(queue_.start != queue_.end) { + output_pixels(&target[queue_.start], queue_.end - queue_.start, collision_identity, queue_.pixel_position, queue_.adder, queue_.reverse_mask); + queue_.start = queue_.end; + } + } + + void enqueue_pixels(const int start, const int end) + { + queue_.start = start; + queue_.end = end; + queue_.pixel_position = pixel_position; + queue_.adder = adder; + queue_.reverse_mask = reverse_mask; + skip_pixels(end - start); + } + + private: + struct QueuedPixels + { + int start, end; + int pixel_position; + int adder; + int reverse_mask; + } queue_; + + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity, int pixel_position, int adder, int reverse_mask) + { + if(pixel_position == 32 || !graphic[graphic_index]) return; int output_cursor = 0; while(pixel_position < 32 && output_cursor < count) { @@ -182,17 +235,11 @@ class TIA { pixel_position += adder; } } - else - { - skip_pixels(count); - } - } - Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0), adder(4) {} } player_[2]; // common actor for things that appear as a horizontal run of pixels - struct HorizontalRun: public Object { + struct HorizontalRun: public Object { int pixel_position; int size; @@ -206,7 +253,7 @@ class TIA { pixel_position = size; } - inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { int output_cursor = 0; while(pixel_position && output_cursor < count) @@ -227,12 +274,12 @@ class TIA { bool locked_to_player; int copy_flags; - inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(!pixel_position) return; if(enabled && !locked_to_player) { - HorizontalRun::enqueue_pixels(target, count, collision_identity); + HorizontalRun::output_pixels(target, count, collision_identity); } else { @@ -249,12 +296,12 @@ class TIA { int enabled_index; const int copy_flags = 0; - inline void enqueue_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) { if(!pixel_position) return; if(enabled[enabled_index]) { - HorizontalRun::enqueue_pixels(target, count, collision_identity); + HorizontalRun::output_pixels(target, count, collision_identity); } else { @@ -272,7 +319,7 @@ class TIA { // drawing methods and state template void draw_object(T &, const uint8_t collision_identity, int start, int end); - template void draw_object_visible(T &, const uint8_t collision_identity, int start, int end); + template void draw_object_visible(T &, const uint8_t collision_identity, int start, int end, int time_now); inline void draw_playfield(int start, int end); inline void output_for_cycles(int number_of_cycles); From c898c8a99ecac70d48983720c0da455ec632b166 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 25 Feb 2017 17:13:22 -0500 Subject: [PATCH 87/91] Ensured the missiles and ball don't attempt to enqueue. Because I don't think they're supposed to. --- Machines/Atari2600/TIA.cpp | 2 +- Machines/Atari2600/TIA.hpp | 24 +++++------------------- 2 files changed, 6 insertions(+), 20 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 0fa51316d..9e56cbd68 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -725,7 +725,7 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi // an appropriate solution would probably be to capture the drawing request into a queue and honour them outside // this loop, clipped to the real output parameters. Assuming all state consumed by draw_pixels is captured, // and mutated now then also queueing resets and skips shouldn't be necessary. - if(next_event_time > time_now) + if(object.enqueues && next_event_time > time_now) { if(start < time_now) { diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 4114a44be..4a1ecaa32 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -146,25 +146,6 @@ class TIA { bool is_moving; Object() : is_moving(false) {}; - - void dequeue_pixels(uint8_t *const target, const uint8_t collision_identity, const int time_now) - { -// if(enqueued_start_ != enqueued_end_) -// { -// static_cast(this)->output_pixels(&target[enqueued_start_], enqueued_end_ - enqueued_start_, collision_identity); -// enqueued_end_ = enqueued_start_ = 0; -// } - } - - void enqueue_pixels(const int start, const int end) - { -// enqueued_start_ = start; -// enqueued_end_ = end; - static_cast(this)->skip_pixels(end - start); - } - - private: -// int enqueued_start_, enqueued_end_; }; // player state @@ -178,6 +159,7 @@ class TIA { int graphic_index; int pixel_position; + const bool enqueues = true; inline void skip_pixels(const int count) { @@ -242,6 +224,7 @@ class TIA { struct HorizontalRun: public Object { int pixel_position; int size; + const bool enqueues = false; inline void skip_pixels(const int count) { @@ -264,6 +247,9 @@ class TIA { } } + void dequeue_pixels(uint8_t *const target, const uint8_t collision_identity, const int time_now) {} + void enqueue_pixels(const int start, const int end) {} + HorizontalRun() : pixel_position(0), size(1) {} }; From e61e355251a37ead6ebd9d507d4c4d8bca1b544a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 25 Feb 2017 17:25:10 -0500 Subject: [PATCH 88/91] Moved to the maximum possibly required queue length of 4. Though the emulated 2600 should never need more than 2 slots as per the current calling pattern, it's not a contractual guarantee. --- Machines/Atari2600/TIA.hpp | 34 ++++++++++++++++++++++++---------- 1 file changed, 24 insertions(+), 10 deletions(-) diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 4a1ecaa32..b8cf5cf76 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -150,7 +150,7 @@ class TIA { // player state struct Player: public Object { - Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0), adder(4) {} + Player() : copy_flags(0), graphic{0, 0}, reverse_mask(false), pixel_position(32), graphic_index(0), adder(4), queue_read_pointer_(0), queue_write_pointer_(0) {} int adder; int copy_flags; // a bit field, corresponding to the first few values of NUSIZ @@ -179,20 +179,33 @@ class TIA { void dequeue_pixels(uint8_t *const target, const uint8_t collision_identity, const int time_now) { - if(queue_.start != queue_.end) + while(queue_read_pointer_ != queue_write_pointer_) { - output_pixels(&target[queue_.start], queue_.end - queue_.start, collision_identity, queue_.pixel_position, queue_.adder, queue_.reverse_mask); - queue_.start = queue_.end; + uint8_t *const start_ptr = &target[queue_[queue_read_pointer_].start]; + if(queue_[queue_read_pointer_].end > time_now) + { + const int length = time_now - queue_[queue_read_pointer_].start; + output_pixels(start_ptr, length, collision_identity, queue_[queue_read_pointer_].pixel_position, queue_[queue_read_pointer_].adder, queue_[queue_read_pointer_].reverse_mask); + queue_[queue_read_pointer_].pixel_position += length * queue_[queue_read_pointer_].adder; + queue_[queue_read_pointer_].start = time_now; + return; + } + else + { + output_pixels(start_ptr, queue_[queue_read_pointer_].end - queue_[queue_read_pointer_].start, collision_identity, queue_[queue_read_pointer_].pixel_position, queue_[queue_read_pointer_].adder, queue_[queue_read_pointer_].reverse_mask); + } + queue_read_pointer_ = (queue_read_pointer_ + 1)&3; } } void enqueue_pixels(const int start, const int end) { - queue_.start = start; - queue_.end = end; - queue_.pixel_position = pixel_position; - queue_.adder = adder; - queue_.reverse_mask = reverse_mask; + queue_[queue_write_pointer_].start = start; + queue_[queue_write_pointer_].end = end; + queue_[queue_write_pointer_].pixel_position = pixel_position; + queue_[queue_write_pointer_].adder = adder; + queue_[queue_write_pointer_].reverse_mask = reverse_mask; + queue_write_pointer_ = (queue_write_pointer_ + 1)&3; skip_pixels(end - start); } @@ -203,7 +216,8 @@ class TIA { int pixel_position; int adder; int reverse_mask; - } queue_; + } queue_[4]; + int queue_read_pointer_, queue_write_pointer_; inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity, int pixel_position, int adder, int reverse_mask) { From 98376de9ad6c43ef6eea1276705015b4b437b2e4 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 25 Feb 2017 22:58:58 -0500 Subject: [PATCH 89/91] Started returning 'no effect' for pot ports, rather than doing nothing. Still very much TODO though. --- Machines/Atari2600/Atari2600.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/Machines/Atari2600/Atari2600.cpp b/Machines/Atari2600/Atari2600.cpp index 25a8b9792..50745cb9b 100644 --- a/Machines/Atari2600/Atari2600.cpp +++ b/Machines/Atari2600/Atari2600.cpp @@ -121,6 +121,7 @@ unsigned int Machine::perform_bus_operation(CPU6502::BusOperation operation, uin case 0x0a: case 0x0b: // TODO: pot ports + returnValue &= 0; break; case 0x0c: From 440467ea3e9ed8e12589fd0a17a839c76d23b107 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 26 Feb 2017 13:39:25 -0500 Subject: [PATCH 90/91] Started communicating which copy is being requested. --- Machines/Atari2600/TIA.cpp | 16 ++++++++++------ Machines/Atari2600/TIA.hpp | 4 ++-- 2 files changed, 12 insertions(+), 8 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 9e56cbd68..262201d55 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -381,7 +381,7 @@ void TIA::set_ball_position() ball_.position = 0; // setting the ball position also triggers a draw - ball_.reset_pixels(); + ball_.reset_pixels(0); } void TIA::set_ball_motion(uint8_t motion) @@ -638,10 +638,10 @@ template void TIA::perform_motion_step(T &object) object.is_moving = false; else { - if(object.position == 159) object.reset_pixels(); - else if(object.position == 15 && object.copy_flags&1) object.reset_pixels(); - else if(object.position == 31 && object.copy_flags&2) object.reset_pixels(); - else if(object.position == 63 && object.copy_flags&4) object.reset_pixels(); + if(object.position == 159) object.reset_pixels(0); + else if(object.position == 15 && object.copy_flags&1) object.reset_pixels(1); + else if(object.position == 31 && object.copy_flags&2) object.reset_pixels(2); + else if(object.position == 63 && object.copy_flags&4) object.reset_pixels(3); else object.skip_pixels(1); object.position = (object.position + 1) % 160; object.motion_step --; @@ -701,17 +701,21 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi // is the next event a graphics trigger? int next_copy = 160; + int next_copy_id = 0; if(object.copy_flags) { if(object.position < 16 && object.copy_flags&1) { next_copy = 16; + next_copy_id = 1; } else if(object.position < 32 && object.copy_flags&2) { next_copy = 32; + next_copy_id = 2; } else if(object.position < 64 && object.copy_flags&4) { next_copy = 64; + next_copy_id = 3; } } @@ -755,7 +759,7 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi // if it's a draw trigger, trigger a draw else if(start == next_copy_time) { - object.reset_pixels(); + object.reset_pixels(next_copy_id); } } } diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index b8cf5cf76..90148258d 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -166,7 +166,7 @@ class TIA { pixel_position = std::min(32, pixel_position + count * adder); } - inline void reset_pixels() + inline void reset_pixels(int copy) { pixel_position = 0; } @@ -245,7 +245,7 @@ class TIA { pixel_position = std::max(0, pixel_position - count); } - inline void reset_pixels() + inline void reset_pixels(int copy) { pixel_position = size; } From e6a84fd26bad3880658c0f1698f20ce2f9216443 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 26 Feb 2017 15:12:31 -0500 Subject: [PATCH 91/91] Attempted a hardware-correct implementation of missile-to-player latching. This completes the last of the **knowing** inaccuracies. The rest are as-of-yet unwitting. --- Machines/Atari2600/TIA.cpp | 41 +++++++++++++++++++++++++------------- Machines/Atari2600/TIA.hpp | 37 +++++++++++++++++++++------------- 2 files changed, 50 insertions(+), 28 deletions(-) diff --git a/Machines/Atari2600/TIA.cpp b/Machines/Atari2600/TIA.cpp index 262201d55..15fb2e939 100644 --- a/Machines/Atari2600/TIA.cpp +++ b/Machines/Atari2600/TIA.cpp @@ -354,10 +354,8 @@ void TIA::set_missile_position(int missile) void TIA::set_missile_position_to_player(int missile, bool lock) { assert(missile >= 0 && missile < 2); - // TODO: implement this correctly; should be triggered by player counter hitting the appropriate point, and - // use additional storage position for enabled - if(missile_[missile].locked_to_player && !lock) missile_[missile].position = player_[missile].position + 1 + 16/player_[missile].adder; missile_[missile].locked_to_player = lock; + player_[missile].latched_pixel4_time = -1; } void TIA::set_missile_motion(int missile, uint8_t motion) @@ -450,8 +448,8 @@ void TIA::output_for_cycles(int number_of_cycles) draw_playfield(latent_start, latent_end); draw_object(player_[0], (uint8_t)CollisionType::Player0, output_cursor, horizontal_counter_); draw_object(player_[1], (uint8_t)CollisionType::Player1, output_cursor, horizontal_counter_); - draw_object(missile_[0], (uint8_t)CollisionType::Missile0, output_cursor, horizontal_counter_); - draw_object(missile_[1], (uint8_t)CollisionType::Missile1, output_cursor, horizontal_counter_); + draw_missile(missile_[0], player_[0], (uint8_t)CollisionType::Missile0, output_cursor, horizontal_counter_); + draw_missile(missile_[1], player_[1], (uint8_t)CollisionType::Missile1, output_cursor, horizontal_counter_); draw_object(ball_, (uint8_t)CollisionType::Ball, output_cursor, horizontal_counter_); // convert to television signals @@ -642,7 +640,7 @@ template void TIA::perform_motion_step(T &object) else if(object.position == 15 && object.copy_flags&1) object.reset_pixels(1); else if(object.position == 31 && object.copy_flags&2) object.reset_pixels(2); else if(object.position == 63 && object.copy_flags&4) object.reset_pixels(3); - else object.skip_pixels(1); + else object.skip_pixels(1, object.motion_time); object.position = (object.position + 1) % 160; object.motion_step --; object.motion_time += 4; @@ -725,25 +723,23 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi // the decision is to progress by length const int length = next_event_time - start; - // TODO: the problem with this is that it uses the enabled/pixel state of each object four cycles early; - // an appropriate solution would probably be to capture the drawing request into a queue and honour them outside - // this loop, clipped to the real output parameters. Assuming all state consumed by draw_pixels is captured, - // and mutated now then also queueing resets and skips shouldn't be necessary. + // enqueue a future intention to draw pixels if spitting them out now would violate accuracy; + // otherwise draw them now if(object.enqueues && next_event_time > time_now) { if(start < time_now) { - object.output_pixels(&collision_buffer_[start], time_now - start, collision_identity); - object.enqueue_pixels(time_now, next_event_time); + object.output_pixels(&collision_buffer_[start], time_now - start, collision_identity, start + first_pixel_cycle - 4); + object.enqueue_pixels(time_now, next_event_time, time_now + first_pixel_cycle - 4); } else { - object.enqueue_pixels(start, next_event_time); + object.enqueue_pixels(start, next_event_time, start + first_pixel_cycle - 4); } } else { - object.output_pixels(&collision_buffer_[start], length, collision_identity); + object.output_pixels(&collision_buffer_[start], length, collision_identity, start + first_pixel_cycle - 4); } // the next interesting event is after next_event_time cycles, so progress @@ -763,3 +759,20 @@ template void TIA::draw_object_visible(T &object, const uint8_t collisi } } } + +#pragma mark - Missile drawing + +void TIA::draw_missile(Missile &missile, Player &player, const uint8_t collision_identity, int start, int end) +{ + if(!missile.locked_to_player || player.latched_pixel4_time < 0) + { + draw_object(missile, collision_identity, start, end); + } + else + { + draw_object(missile, collision_identity, start, player.latched_pixel4_time); + missile.position = 0; + draw_object(missile, collision_identity, player.latched_pixel4_time, end); + player.latched_pixel4_time = -1; + } +} diff --git a/Machines/Atari2600/TIA.hpp b/Machines/Atari2600/TIA.hpp index 90148258d..a462fb54f 100644 --- a/Machines/Atari2600/TIA.hpp +++ b/Machines/Atari2600/TIA.hpp @@ -159,22 +159,29 @@ class TIA { int graphic_index; int pixel_position; + int latched_pixel4_time; const bool enqueues = true; - inline void skip_pixels(const int count) + inline void skip_pixels(const int count, int from_horizontal_counter) { + int old_pixel_position = pixel_position; pixel_position = std::min(32, pixel_position + count * adder); + if(!copy_index_ && old_pixel_position < 16 && pixel_position >= 16) + { + latched_pixel4_time = from_horizontal_counter + (16 - old_pixel_position) / adder; + } } inline void reset_pixels(int copy) { pixel_position = 0; + copy_index_ = copy; } - inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity, int from_horizontal_counter) { output_pixels(target, count, collision_identity, pixel_position, adder, reverse_mask); - skip_pixels(count); + skip_pixels(count, from_horizontal_counter); } void dequeue_pixels(uint8_t *const target, const uint8_t collision_identity, const int time_now) @@ -198,7 +205,7 @@ class TIA { } } - void enqueue_pixels(const int start, const int end) + void enqueue_pixels(const int start, const int end, int from_horizontal_counter) { queue_[queue_write_pointer_].start = start; queue_[queue_write_pointer_].end = end; @@ -206,10 +213,11 @@ class TIA { queue_[queue_write_pointer_].adder = adder; queue_[queue_write_pointer_].reverse_mask = reverse_mask; queue_write_pointer_ = (queue_write_pointer_ + 1)&3; - skip_pixels(end - start); + skip_pixels(end - start, from_horizontal_counter); } private: + int copy_index_; struct QueuedPixels { int start, end; @@ -240,7 +248,7 @@ class TIA { int size; const bool enqueues = false; - inline void skip_pixels(const int count) + inline void skip_pixels(const int count, int from_horizontal_counter) { pixel_position = std::max(0, pixel_position - count); } @@ -250,7 +258,7 @@ class TIA { pixel_position = size; } - inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity, int from_horizontal_counter) { int output_cursor = 0; while(pixel_position && output_cursor < count) @@ -262,7 +270,7 @@ class TIA { } void dequeue_pixels(uint8_t *const target, const uint8_t collision_identity, const int time_now) {} - void enqueue_pixels(const int start, const int end) {} + void enqueue_pixels(const int start, const int end, int from_horizontal_counter) {} HorizontalRun() : pixel_position(0), size(1) {} }; @@ -274,16 +282,16 @@ class TIA { bool locked_to_player; int copy_flags; - inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity, int from_horizontal_counter) { if(!pixel_position) return; if(enabled && !locked_to_player) { - HorizontalRun::output_pixels(target, count, collision_identity); + HorizontalRun::output_pixels(target, count, collision_identity, from_horizontal_counter); } else { - skip_pixels(count); + skip_pixels(count, from_horizontal_counter); } } @@ -296,16 +304,16 @@ class TIA { int enabled_index; const int copy_flags = 0; - inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity) + inline void output_pixels(uint8_t *const target, const int count, const uint8_t collision_identity, int from_horizontal_counter) { if(!pixel_position) return; if(enabled[enabled_index]) { - HorizontalRun::output_pixels(target, count, collision_identity); + HorizontalRun::output_pixels(target, count, collision_identity, from_horizontal_counter); } else { - skip_pixels(count); + skip_pixels(count, from_horizontal_counter); } } @@ -318,6 +326,7 @@ class TIA { template void perform_motion_step(T &object); // drawing methods and state + void draw_missile(Missile &, Player &, const uint8_t collision_identity, int start, int end); template void draw_object(T &, const uint8_t collision_identity, int start, int end); template void draw_object_visible(T &, const uint8_t collision_identity, int start, int end, int time_now); inline void draw_playfield(int start, int end);