1
0
mirror of https://github.com/TomHarte/CLK.git synced 2025-01-12 15:31:09 +00:00

Reorganised state, with an eye towards unifying object motion and triggers.

This commit is contained in:
Thomas Harte 2017-02-20 17:58:28 -05:00
parent 87afa9140e
commit 57f434c199
2 changed files with 58 additions and 43 deletions

View File

@ -26,9 +26,6 @@ TIA::TIA(bool create_crt) :
pixel_target_(nullptr), pixel_target_(nullptr),
background_{0, 0}, 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_blank_extend_(false),
collision_flags_(0) 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 // 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 // code is written to start a draw upon wraparound from 159 to 0, so -1 is the
// correct option rather than 159. // 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) 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) 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() void TIA::set_ball_position()
{ {
position_[(int)MotionIndex::Ball] = 0; object_[(int)MotionIndex::Ball].position = 0;
// setting the ball position also triggers a draw // setting the ball position also triggers a draw
ball_.pixel_position = ball_.size; ball_.pixel_position = ball_.size;
@ -379,20 +376,20 @@ void TIA::set_ball_position()
void TIA::set_ball_motion(uint8_t motion) 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() void TIA::move()
{ {
horizontal_blank_extend_ = true; horizontal_blank_extend_ = true;
is_moving_[0] = is_moving_[1] = is_moving_[2] = is_moving_[3] = is_moving_[4] = true; object_[0].is_moving = object_[1].is_moving = object_[2].is_moving = object_[3].is_moving = object_[4].is_moving = true;
motion_step_[0] = motion_step_[1] = motion_step_[2] = motion_step_[3] = motion_step_[4] = 15; object_[0].motion_step = object_[1].motion_step = object_[2].motion_step = object_[3].motion_step = object_[4].motion_step = 15;
motion_time_[0] = motion_time_[1] = motion_time_[2] = motion_time_[3] = motion_time_[4] = (horizontal_counter_ + 3) & ~3; 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() 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) 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()); if(line_end_function_) line_end_function_(collision_buffer_.data());
memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_) memset(collision_buffer_.data(), 0, 160); // sizeof(collision_buffer_)
horizontal_blank_extend_ = false; 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 // 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) void TIA::perform_motion_step(int identity)
{ {
if((motion_step_[identity] ^ (motion_[identity] ^ 8)) == 0xf) Object &object = object_[identity];
is_moving_[identity] = false; if((object.motion_step ^ (object.motion ^ 8)) == 0xf)
object.is_moving = false;
else else
{ {
position_[identity] ++; object.position ++;
motion_step_[identity] --; object.motion_step --;
motion_time_[identity] += 4; object.motion_time += 4;
} }
} }
int TIA::perform_border_motion(int identity, int start, int end) 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; int steps_taken = 0;
while(is_moving_[identity] && motion_time_[identity] < end) while(object.is_moving && object.motion_time < end)
{ {
perform_motion_step(identity); perform_motion_step(identity);
steps_taken++; steps_taken++;
} }
position_[identity] %= 160; object.position %= 160;
return steps_taken; 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) 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; int adder = 4 >> player.size;
// perform a miniature event loop on (i) triggering draws; (ii) drawing; and (iii) motion // 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) while(start < end)
{ {
int next_event_time = end; int next_event_time = end;
// is the next event a movement tick? // 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; next_event_time = next_motion_time;
} }
// is the next event a graphics trigger? // is the next event a graphics trigger?
int next_copy = 160; int next_copy = 160;
if(position < 16 && player.copy_flags&1) if(object.position < 16 && player.copy_flags&1)
{ {
next_copy = 16; next_copy = 16;
} else if(position < 32 && player.copy_flags&2) } else if(object.position < 32 && player.copy_flags&2)
{ {
next_copy = 32; next_copy = 32;
} else if(position < 64 && player.copy_flags&4) } else if(object.position < 64 && player.copy_flags&4)
{ {
next_copy = 64; 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; if(next_copy_time < next_event_time) next_event_time = next_copy_time;
// the decision is to progress by length // 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 // 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; start = next_event_time;
// if the event is a motion tick, apply // 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); perform_motion_step(position_identity);
next_motion_time += 4; next_motion_time += 4;
@ -746,7 +745,7 @@ void TIA::draw_player(Player &player, CollisionType collision_identity, const in
} }
// move further if required // 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); perform_motion_step(position_identity);
player.pixel_position = std::min(32, player.pixel_position + adder); 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 // 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); perform_motion_step((int)MotionIndex::Ball);
ball_.pixel_position = std::max(0, ball_.pixel_position - 1); 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) 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 // 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) while(start < end)
{ {
int next_event_time = end; int next_event_time = end;
// is the next event a movement tick? // 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; 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? // is the next event a graphics trigger?
if(ball_.enabled[ball_.enabled_index]) 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; int next_copy_time = start + time_until_copy;
if(next_copy_time < next_event_time) next_event_time = next_copy_time; 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 // 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; start = next_event_time;
// if the event is a motion tick, apply // 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); perform_motion_step((int)MotionIndex::Ball);
next_motion_time += 4; next_motion_time += 4;
} }
// if it's a draw trigger, trigger a draw // if it's a draw trigger, trigger a draw
if(!position) if(!object.position)
{ {
ball_.pixel_position = ball_.size; ball_.pixel_position = ball_.size;
} }

View File

@ -166,11 +166,27 @@ class TIA {
// movement // movement
bool horizontal_blank_extend_; bool horizontal_blank_extend_;
int motion_[5]; struct Object {
int motion_step_[5]; // the two programmer-set values
int motion_time_[5]; int position;
int position_[5]; int motion;
bool is_moving_[5];
// 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 { enum class MotionIndex : uint8_t {
Ball, Ball,
Player0, Player0,