2015-07-16 19:56:02 -04:00
|
|
|
//
|
|
|
|
// Atari2600.cpp
|
2015-07-26 15:25:11 -04:00
|
|
|
// CLK
|
2015-07-16 19:56:02 -04:00
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 14/07/2015.
|
|
|
|
// Copyright © 2015 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "Atari2600.hpp"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2017-03-18 21:01:58 -04:00
|
|
|
#include "Cartridges/CartridgeAtari8k.hpp"
|
|
|
|
#include "Cartridges/CartridgeAtari16k.hpp"
|
|
|
|
#include "Cartridges/CartridgeAtari32k.hpp"
|
|
|
|
#include "Cartridges/CartridgeActivisionStack.hpp"
|
|
|
|
#include "Cartridges/CartridgeCBSRAMPlus.hpp"
|
|
|
|
#include "Cartridges/CartridgeCommaVid.hpp"
|
|
|
|
#include "Cartridges/CartridgeMegaBoy.hpp"
|
|
|
|
#include "Cartridges/CartridgeMNetwork.hpp"
|
|
|
|
#include "Cartridges/CartridgeParkerBros.hpp"
|
2017-03-18 22:08:47 -04:00
|
|
|
#include "Cartridges/CartridgePitfall2.hpp"
|
2017-03-18 21:01:58 -04:00
|
|
|
#include "Cartridges/CartridgeTigervision.hpp"
|
|
|
|
#include "Cartridges/CartridgeUnpaged.hpp"
|
2017-03-18 14:01:04 -04:00
|
|
|
|
2015-07-16 19:56:02 -04:00
|
|
|
using namespace Atari2600;
|
2016-05-16 08:01:29 -04:00
|
|
|
namespace {
|
2016-08-14 13:33:20 -04:00
|
|
|
static const double NTSC_clock_rate = 1194720;
|
|
|
|
static const double PAL_clock_rate = 1182298;
|
2016-05-16 08:01:29 -04:00
|
|
|
}
|
2015-07-16 19:56:02 -04:00
|
|
|
|
2015-12-06 16:53:37 -05:00
|
|
|
Machine::Machine() :
|
2017-02-19 21:20:37 -05:00
|
|
|
frame_record_pointer_(0),
|
2017-03-14 20:07:54 -04:00
|
|
|
is_ntsc_(true) {
|
2016-08-14 13:33:20 -04:00
|
|
|
set_clock_rate(NTSC_clock_rate);
|
2016-04-24 06:56:08 -04:00
|
|
|
}
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
void Machine::setup_output(float aspect_ratio) {
|
2017-03-18 14:01:04 -04:00
|
|
|
bus_->tia_.reset(new TIA);
|
|
|
|
bus_->speaker_.reset(new Speaker);
|
2017-04-03 21:16:39 -04:00
|
|
|
bus_->speaker_->set_input_rate((float)(get_clock_rate() / (double)CPUTicksPerAudioTick));
|
2017-03-18 14:01:04 -04:00
|
|
|
bus_->tia_->get_crt()->set_delegate(this);
|
2016-05-11 21:07:18 -04:00
|
|
|
}
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
void Machine::close_output() {
|
2017-03-18 14:01:04 -04:00
|
|
|
bus_.reset();
|
2016-04-24 20:34:25 -04:00
|
|
|
}
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
Machine::~Machine() {
|
2016-04-24 20:34:25 -04:00
|
|
|
close_output();
|
2015-07-27 21:15:10 -04:00
|
|
|
}
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
void Machine::set_digital_input(Atari2600DigitalInput input, bool state) {
|
2015-08-18 20:33:24 -04:00
|
|
|
switch (input) {
|
2017-03-18 14:01:04 -04:00
|
|
|
case Atari2600DigitalInputJoy1Up: bus_->mos6532_.update_port_input(0, 0x10, state); break;
|
|
|
|
case Atari2600DigitalInputJoy1Down: bus_->mos6532_.update_port_input(0, 0x20, state); break;
|
|
|
|
case Atari2600DigitalInputJoy1Left: bus_->mos6532_.update_port_input(0, 0x40, state); break;
|
|
|
|
case Atari2600DigitalInputJoy1Right: bus_->mos6532_.update_port_input(0, 0x80, state); break;
|
2016-06-19 18:57:40 -04:00
|
|
|
|
2017-03-18 14:01:04 -04:00
|
|
|
case Atari2600DigitalInputJoy2Up: bus_->mos6532_.update_port_input(0, 0x01, state); break;
|
|
|
|
case Atari2600DigitalInputJoy2Down: bus_->mos6532_.update_port_input(0, 0x02, state); break;
|
|
|
|
case Atari2600DigitalInputJoy2Left: bus_->mos6532_.update_port_input(0, 0x04, state); break;
|
|
|
|
case Atari2600DigitalInputJoy2Right: bus_->mos6532_.update_port_input(0, 0x08, state); break;
|
2015-08-18 20:58:05 -04:00
|
|
|
|
|
|
|
// TODO: latching
|
2017-03-18 14:01:04 -04:00
|
|
|
case Atari2600DigitalInputJoy1Fire: if(state) bus_->tia_input_value_[0] &= ~0x80; else bus_->tia_input_value_[0] |= 0x80; break;
|
|
|
|
case Atari2600DigitalInputJoy2Fire: if(state) bus_->tia_input_value_[1] &= ~0x80; else bus_->tia_input_value_[1] |= 0x80; break;
|
2015-08-18 20:33:24 -04:00
|
|
|
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
void Machine::set_switch_is_enabled(Atari2600Switch input, bool state) {
|
2016-06-19 19:36:34 -04:00
|
|
|
switch(input) {
|
2017-03-18 14:01:04 -04:00
|
|
|
case Atari2600SwitchReset: bus_->mos6532_.update_port_input(1, 0x01, state); break;
|
|
|
|
case Atari2600SwitchSelect: bus_->mos6532_.update_port_input(1, 0x02, state); break;
|
|
|
|
case Atari2600SwitchColour: bus_->mos6532_.update_port_input(1, 0x08, state); break;
|
|
|
|
case Atari2600SwitchLeftPlayerDifficulty: bus_->mos6532_.update_port_input(1, 0x40, state); break;
|
|
|
|
case Atari2600SwitchRightPlayerDifficulty: bus_->mos6532_.update_port_input(1, 0x80, state); break;
|
2016-06-19 19:36:34 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
void Machine::configure_as_target(const StaticAnalyser::Target &target) {
|
2017-03-18 18:21:01 -04:00
|
|
|
const std::vector<uint8_t> &rom = target.cartridges.front()->get_segments().front().data;
|
2017-03-18 14:46:46 -04:00
|
|
|
switch(target.atari.paging_model) {
|
2017-03-18 18:56:20 -04:00
|
|
|
case StaticAnalyser::Atari2600PagingModel::ActivisionStack: bus_.reset(new CartridgeActivisionStack(rom)); break;
|
|
|
|
case StaticAnalyser::Atari2600PagingModel::CBSRamPlus: bus_.reset(new CartridgeCBSRAMPlus(rom)); break;
|
2017-03-18 22:08:47 -04:00
|
|
|
case StaticAnalyser::Atari2600PagingModel::CommaVid: bus_.reset(new CartridgeCommaVid(rom)); break;
|
2017-03-18 19:02:34 -04:00
|
|
|
case StaticAnalyser::Atari2600PagingModel::MegaBoy: bus_.reset(new CartridgeMegaBoy(rom)); break;
|
2017-03-18 20:51:49 -04:00
|
|
|
case StaticAnalyser::Atari2600PagingModel::MNetwork: bus_.reset(new CartridgeMNetwork(rom)); break;
|
2017-03-18 22:08:47 -04:00
|
|
|
case StaticAnalyser::Atari2600PagingModel::None: bus_.reset(new CartridgeUnpaged(rom)); break;
|
|
|
|
case StaticAnalyser::Atari2600PagingModel::ParkerBros: bus_.reset(new CartridgeParkerBros(rom)); break;
|
|
|
|
case StaticAnalyser::Atari2600PagingModel::Pitfall2: bus_.reset(new CartridgePitfall2(rom)); break;
|
|
|
|
case StaticAnalyser::Atari2600PagingModel::Tigervision: bus_.reset(new CartridgeTigervision(rom)); break;
|
|
|
|
|
2017-03-18 15:04:01 -04:00
|
|
|
case StaticAnalyser::Atari2600PagingModel::Atari8k:
|
|
|
|
if(target.atari.uses_superchip) {
|
2017-03-18 18:21:01 -04:00
|
|
|
bus_.reset(new CartridgeAtari8kSuperChip(rom));
|
2017-03-18 15:04:01 -04:00
|
|
|
} else {
|
2017-03-18 18:21:01 -04:00
|
|
|
bus_.reset(new CartridgeAtari8k(rom));
|
2017-03-18 15:04:01 -04:00
|
|
|
}
|
|
|
|
break;
|
2017-03-18 17:34:34 -04:00
|
|
|
case StaticAnalyser::Atari2600PagingModel::Atari16k:
|
|
|
|
if(target.atari.uses_superchip) {
|
2017-03-18 18:56:20 -04:00
|
|
|
bus_.reset(new CartridgeAtari16kSuperChip(rom));
|
2017-03-18 17:34:34 -04:00
|
|
|
} else {
|
2017-03-18 18:56:20 -04:00
|
|
|
bus_.reset(new CartridgeAtari16k(rom));
|
2017-03-18 17:34:34 -04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case StaticAnalyser::Atari2600PagingModel::Atari32k:
|
|
|
|
if(target.atari.uses_superchip) {
|
2017-03-18 18:56:20 -04:00
|
|
|
bus_.reset(new CartridgeAtari32kSuperChip(rom));
|
2017-03-18 17:34:34 -04:00
|
|
|
} else {
|
2017-03-18 18:56:20 -04:00
|
|
|
bus_.reset(new CartridgeAtari32k(rom));
|
2017-03-18 17:34:34 -04:00
|
|
|
}
|
|
|
|
break;
|
2017-03-18 14:46:46 -04:00
|
|
|
}
|
2016-06-01 19:27:04 -04:00
|
|
|
}
|
2017-02-19 21:20:37 -05:00
|
|
|
|
|
|
|
#pragma mark - CRT delegate
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
void Machine::crt_did_end_batch_of_frames(Outputs::CRT::CRT *crt, unsigned int number_of_frames, unsigned int number_of_unexpected_vertical_syncs) {
|
2017-02-19 21:20:37 -05:00
|
|
|
const size_t number_of_frame_records = sizeof(frame_records_) / sizeof(frame_records_[0]);
|
2017-03-06 19:37:35 -05:00
|
|
|
frame_records_[frame_record_pointer_ % number_of_frame_records].number_of_frames = number_of_frames;
|
|
|
|
frame_records_[frame_record_pointer_ % number_of_frame_records].number_of_unexpected_vertical_syncs = number_of_unexpected_vertical_syncs;
|
|
|
|
frame_record_pointer_ ++;
|
2017-02-19 21:20:37 -05:00
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
if(frame_record_pointer_ >= 6) {
|
2017-03-06 19:37:35 -05:00
|
|
|
unsigned int total_number_of_frames = 0;
|
|
|
|
unsigned int total_number_of_unexpected_vertical_syncs = 0;
|
2017-03-14 20:07:54 -04:00
|
|
|
for(size_t c = 0; c < number_of_frame_records; c++) {
|
2017-03-06 19:37:35 -05:00
|
|
|
total_number_of_frames += frame_records_[c].number_of_frames;
|
|
|
|
total_number_of_unexpected_vertical_syncs += frame_records_[c].number_of_unexpected_vertical_syncs;
|
2017-02-19 21:20:37 -05:00
|
|
|
}
|
|
|
|
|
2017-03-14 20:07:54 -04:00
|
|
|
if(total_number_of_unexpected_vertical_syncs >= total_number_of_frames >> 1) {
|
|
|
|
for(size_t c = 0; c < number_of_frame_records; c++) {
|
2017-03-06 19:37:35 -05:00
|
|
|
frame_records_[c].number_of_frames = 0;
|
|
|
|
frame_records_[c].number_of_unexpected_vertical_syncs = 0;
|
|
|
|
}
|
|
|
|
is_ntsc_ ^= true;
|
2017-02-19 21:20:37 -05:00
|
|
|
|
2017-03-06 19:37:35 -05:00
|
|
|
double clock_rate;
|
2017-03-14 20:07:54 -04:00
|
|
|
if(is_ntsc_) {
|
2017-03-06 19:37:35 -05:00
|
|
|
clock_rate = NTSC_clock_rate;
|
2017-03-18 14:01:04 -04:00
|
|
|
bus_->tia_->set_output_mode(TIA::OutputMode::NTSC);
|
2017-03-14 20:07:54 -04:00
|
|
|
} else {
|
2017-03-06 19:37:35 -05:00
|
|
|
clock_rate = PAL_clock_rate;
|
2017-03-18 14:01:04 -04:00
|
|
|
bus_->tia_->set_output_mode(TIA::OutputMode::PAL);
|
2017-03-06 19:37:35 -05:00
|
|
|
}
|
|
|
|
|
2017-04-03 21:16:39 -04:00
|
|
|
bus_->speaker_->set_input_rate((float)(clock_rate / (double)CPUTicksPerAudioTick));
|
|
|
|
bus_->speaker_->set_high_frequency_cut_off((float)(clock_rate / ((double)CPUTicksPerAudioTick * 2.0)));
|
2017-03-06 19:37:35 -05:00
|
|
|
set_clock_rate(clock_rate);
|
|
|
|
}
|
2017-02-19 21:20:37 -05:00
|
|
|
}
|
|
|
|
}
|