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.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2015 Thomas Harte. All rights reserved.
|
2015-07-16 19:56:02 -04:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "Atari2600.hpp"
|
2017-11-11 15:28:40 -05:00
|
|
|
|
2015-07-16 19:56:02 -04:00
|
|
|
#include <algorithm>
|
2017-11-11 15:28:40 -05:00
|
|
|
#include <cstdio>
|
2015-07-16 19:56:02 -04:00
|
|
|
|
2018-03-09 15:19:02 -05:00
|
|
|
#include "../CRTMachine.hpp"
|
|
|
|
#include "../JoystickMachine.hpp"
|
|
|
|
|
2018-03-09 16:07:29 -05:00
|
|
|
#include "../../Analyser/Static/Atari/Target.hpp"
|
|
|
|
|
2017-08-16 12:39:15 -04:00
|
|
|
#include "Cartridges/Atari8k.hpp"
|
|
|
|
#include "Cartridges/Atari16k.hpp"
|
|
|
|
#include "Cartridges/Atari32k.hpp"
|
|
|
|
#include "Cartridges/ActivisionStack.hpp"
|
|
|
|
#include "Cartridges/CBSRAMPlus.hpp"
|
|
|
|
#include "Cartridges/CommaVid.hpp"
|
|
|
|
#include "Cartridges/MegaBoy.hpp"
|
|
|
|
#include "Cartridges/MNetwork.hpp"
|
|
|
|
#include "Cartridges/ParkerBros.hpp"
|
|
|
|
#include "Cartridges/Pitfall2.hpp"
|
|
|
|
#include "Cartridges/Tigervision.hpp"
|
|
|
|
#include "Cartridges/Unpaged.hpp"
|
2017-03-18 14:01:04 -04:00
|
|
|
|
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
|
|
|
|
2017-08-16 14:52:40 -04:00
|
|
|
namespace Atari2600 {
|
2016-04-24 06:56:08 -04:00
|
|
|
|
2018-06-11 21:35:03 -04:00
|
|
|
class Joystick: public Inputs::ConcreteJoystick {
|
2017-10-15 20:44:59 -04:00
|
|
|
public:
|
2017-11-11 15:28:40 -05:00
|
|
|
Joystick(Bus *bus, std::size_t shift, std::size_t fire_tia_input) :
|
2018-06-11 21:35:03 -04:00
|
|
|
ConcreteJoystick({
|
2018-06-10 20:45:52 -04:00
|
|
|
Input(Input::Up),
|
|
|
|
Input(Input::Down),
|
|
|
|
Input(Input::Left),
|
|
|
|
Input(Input::Right),
|
|
|
|
Input(Input::Fire)
|
2018-06-11 21:35:03 -04:00
|
|
|
}),
|
|
|
|
bus_(bus), shift_(shift), fire_tia_input_(fire_tia_input) {}
|
2018-02-25 19:08:50 -05:00
|
|
|
|
2018-06-11 21:35:03 -04:00
|
|
|
void did_set_input(const Input &digital_input, bool is_active) override {
|
2018-02-25 19:08:50 -05:00
|
|
|
switch(digital_input.type) {
|
2018-06-10 20:45:52 -04:00
|
|
|
case Input::Up: bus_->mos6532_.update_port_input(0, 0x10 >> shift_, is_active); break;
|
|
|
|
case Input::Down: bus_->mos6532_.update_port_input(0, 0x20 >> shift_, is_active); break;
|
|
|
|
case Input::Left: bus_->mos6532_.update_port_input(0, 0x40 >> shift_, is_active); break;
|
|
|
|
case Input::Right: bus_->mos6532_.update_port_input(0, 0x80 >> shift_, is_active); break;
|
2017-10-15 20:44:59 -04:00
|
|
|
|
|
|
|
// TODO: latching
|
2018-06-10 20:45:52 -04:00
|
|
|
case Input::Fire:
|
2017-10-15 20:44:59 -04:00
|
|
|
if(is_active)
|
|
|
|
bus_->tia_input_value_[fire_tia_input_] &= ~0x80;
|
|
|
|
else
|
|
|
|
bus_->tia_input_value_[fire_tia_input_] |= 0x80;
|
|
|
|
break;
|
2018-02-25 19:08:50 -05:00
|
|
|
|
|
|
|
default: break;
|
2017-10-15 20:44:59 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Bus *bus_;
|
2017-11-11 15:28:40 -05:00
|
|
|
std::size_t shift_, fire_tia_input_;
|
2017-10-15 20:44:59 -04:00
|
|
|
};
|
|
|
|
|
2017-08-16 14:52:40 -04:00
|
|
|
class ConcreteMachine:
|
|
|
|
public Machine,
|
2018-03-09 15:19:02 -05:00
|
|
|
public CRTMachine::Machine,
|
|
|
|
public JoystickMachine::Machine,
|
2017-08-16 14:52:40 -04:00
|
|
|
public Outputs::CRT::Delegate {
|
|
|
|
public:
|
2018-07-10 20:00:46 -04:00
|
|
|
ConcreteMachine(const Analyser::Static::Atari::Target &target) {
|
2017-08-16 14:52:40 -04:00
|
|
|
set_clock_rate(NTSC_clock_rate);
|
2016-05-11 21:07:18 -04:00
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
const std::vector<uint8_t> &rom = target.media.cartridges.front()->get_segments().front().data;
|
2018-03-09 16:07:29 -05:00
|
|
|
|
|
|
|
using PagingModel = Analyser::Static::Atari::Target::PagingModel;
|
2018-07-10 20:00:46 -04:00
|
|
|
switch(target.paging_model) {
|
2018-03-09 16:07:29 -05:00
|
|
|
case PagingModel::ActivisionStack: bus_.reset(new Cartridge::Cartridge<Cartridge::ActivisionStack>(rom)); break;
|
|
|
|
case PagingModel::CBSRamPlus: bus_.reset(new Cartridge::Cartridge<Cartridge::CBSRAMPlus>(rom)); break;
|
|
|
|
case PagingModel::CommaVid: bus_.reset(new Cartridge::Cartridge<Cartridge::CommaVid>(rom)); break;
|
|
|
|
case PagingModel::MegaBoy: bus_.reset(new Cartridge::Cartridge<Cartridge::MegaBoy>(rom)); break;
|
|
|
|
case PagingModel::MNetwork: bus_.reset(new Cartridge::Cartridge<Cartridge::MNetwork>(rom)); break;
|
|
|
|
case PagingModel::None: bus_.reset(new Cartridge::Cartridge<Cartridge::Unpaged>(rom)); break;
|
|
|
|
case PagingModel::ParkerBros: bus_.reset(new Cartridge::Cartridge<Cartridge::ParkerBros>(rom)); break;
|
|
|
|
case PagingModel::Pitfall2: bus_.reset(new Cartridge::Cartridge<Cartridge::Pitfall2>(rom)); break;
|
|
|
|
case PagingModel::Tigervision: bus_.reset(new Cartridge::Cartridge<Cartridge::Tigervision>(rom)); break;
|
|
|
|
|
|
|
|
case PagingModel::Atari8k:
|
2018-07-10 20:00:46 -04:00
|
|
|
if(target.uses_superchip) {
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_.reset(new Cartridge::Cartridge<Cartridge::Atari8kSuperChip>(rom));
|
|
|
|
} else {
|
|
|
|
bus_.reset(new Cartridge::Cartridge<Cartridge::Atari8k>(rom));
|
|
|
|
}
|
|
|
|
break;
|
2018-03-09 16:07:29 -05:00
|
|
|
case PagingModel::Atari16k:
|
2018-07-10 20:00:46 -04:00
|
|
|
if(target.uses_superchip) {
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_.reset(new Cartridge::Cartridge<Cartridge::Atari16kSuperChip>(rom));
|
|
|
|
} else {
|
|
|
|
bus_.reset(new Cartridge::Cartridge<Cartridge::Atari16k>(rom));
|
|
|
|
}
|
|
|
|
break;
|
2018-03-09 16:07:29 -05:00
|
|
|
case PagingModel::Atari32k:
|
2018-07-10 20:00:46 -04:00
|
|
|
if(target.uses_superchip) {
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_.reset(new Cartridge::Cartridge<Cartridge::Atari32kSuperChip>(rom));
|
|
|
|
} else {
|
|
|
|
bus_.reset(new Cartridge::Cartridge<Cartridge::Atari32k>(rom));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-10-15 20:44:59 -04:00
|
|
|
|
2017-10-15 20:49:47 -04:00
|
|
|
joysticks_.emplace_back(new Joystick(bus_.get(), 0, 0));
|
|
|
|
joysticks_.emplace_back(new Joystick(bus_.get(), 4, 1));
|
2017-08-16 14:52:40 -04:00
|
|
|
}
|
2015-07-27 21:15:10 -04:00
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
~ConcreteMachine() {
|
|
|
|
close_output();
|
2017-08-17 10:48:29 -04:00
|
|
|
}
|
|
|
|
|
2017-10-15 20:49:47 -04:00
|
|
|
std::vector<std::unique_ptr<Inputs::Joystick>> &get_joysticks() override {
|
2017-10-15 20:44:59 -04:00
|
|
|
return joysticks_;
|
2017-08-16 14:52:40 -04:00
|
|
|
}
|
2015-08-18 20:33:24 -04:00
|
|
|
|
2017-10-15 20:44:59 -04:00
|
|
|
void set_switch_is_enabled(Atari2600Switch input, bool state) override {
|
2017-08-16 14:52:40 -04:00
|
|
|
switch(input) {
|
|
|
|
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;
|
2017-03-18 17:34:34 -04:00
|
|
|
}
|
2017-08-16 14:52:40 -04:00
|
|
|
}
|
2017-02-19 21:20:37 -05:00
|
|
|
|
2018-02-14 21:46:50 -05:00
|
|
|
bool get_switch_is_enabled(Atari2600Switch input) override {
|
|
|
|
uint8_t port_input = bus_->mos6532_.get_port_input(1);
|
|
|
|
switch(input) {
|
|
|
|
case Atari2600SwitchReset: return !!(port_input & 0x01);
|
|
|
|
case Atari2600SwitchSelect: return !!(port_input & 0x02);
|
|
|
|
case Atari2600SwitchColour: return !!(port_input & 0x08);
|
|
|
|
case Atari2600SwitchLeftPlayerDifficulty: return !!(port_input & 0x40);
|
|
|
|
case Atari2600SwitchRightPlayerDifficulty: return !!(port_input & 0x80);
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-15 20:44:59 -04:00
|
|
|
void set_reset_switch(bool state) override {
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_->set_reset_line(state);
|
|
|
|
}
|
2017-02-19 21:20:37 -05:00
|
|
|
|
2017-08-16 14:52:40 -04:00
|
|
|
// to satisfy CRTMachine::Machine
|
2017-10-15 20:44:59 -04:00
|
|
|
void setup_output(float aspect_ratio) override {
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_->tia_.reset(new TIA);
|
2017-12-17 21:26:06 -05:00
|
|
|
bus_->speaker_.set_input_rate(static_cast<float>(get_clock_rate() / static_cast<double>(CPUTicksPerAudioTick)));
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_->tia_->get_crt()->set_delegate(this);
|
|
|
|
}
|
2017-02-19 21:20:37 -05:00
|
|
|
|
2017-10-15 20:44:59 -04:00
|
|
|
void close_output() override {
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_.reset();
|
2017-02-19 21:20:37 -05:00
|
|
|
}
|
|
|
|
|
2017-12-17 21:26:06 -05:00
|
|
|
Outputs::CRT::CRT *get_crt() override {
|
2017-08-16 14:52:40 -04:00
|
|
|
return bus_->tia_->get_crt();
|
|
|
|
}
|
|
|
|
|
2017-12-17 21:26:06 -05:00
|
|
|
Outputs::Speaker::Speaker *get_speaker() override {
|
|
|
|
return &bus_->speaker_;
|
2017-08-16 14:52:40 -04:00
|
|
|
}
|
2017-03-06 19:37:35 -05:00
|
|
|
|
2017-10-15 20:44:59 -04:00
|
|
|
void run_for(const Cycles cycles) override {
|
2017-08-16 14:52:40 -04:00
|
|
|
bus_->run_for(cycles);
|
2018-03-07 14:26:07 -05:00
|
|
|
bus_->apply_confidence(confidence_counter_);
|
2017-03-06 19:37:35 -05:00
|
|
|
}
|
2017-08-16 14:52:40 -04:00
|
|
|
|
|
|
|
// to satisfy Outputs::CRT::Delegate
|
2017-10-15 20:44:59 -04:00
|
|
|
void crt_did_end_batch_of_frames(Outputs::CRT::CRT *crt, unsigned int number_of_frames, unsigned int number_of_unexpected_vertical_syncs) override {
|
2017-11-11 15:28:40 -05:00
|
|
|
const std::size_t number_of_frame_records = sizeof(frame_records_) / sizeof(frame_records_[0]);
|
2017-08-16 14:52:40 -04: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_ ++;
|
|
|
|
|
|
|
|
if(frame_record_pointer_ >= 6) {
|
|
|
|
unsigned int total_number_of_frames = 0;
|
|
|
|
unsigned int total_number_of_unexpected_vertical_syncs = 0;
|
2017-11-11 15:28:40 -05:00
|
|
|
for(std::size_t c = 0; c < number_of_frame_records; c++) {
|
2017-08-16 14:52:40 -04: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;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(total_number_of_unexpected_vertical_syncs >= total_number_of_frames >> 1) {
|
2017-11-11 15:28:40 -05:00
|
|
|
for(std::size_t c = 0; c < number_of_frame_records; c++) {
|
2017-08-16 14:52:40 -04:00
|
|
|
frame_records_[c].number_of_frames = 0;
|
|
|
|
frame_records_[c].number_of_unexpected_vertical_syncs = 0;
|
|
|
|
}
|
|
|
|
is_ntsc_ ^= true;
|
|
|
|
|
|
|
|
double clock_rate;
|
|
|
|
if(is_ntsc_) {
|
|
|
|
clock_rate = NTSC_clock_rate;
|
|
|
|
bus_->tia_->set_output_mode(TIA::OutputMode::NTSC);
|
|
|
|
} else {
|
|
|
|
clock_rate = PAL_clock_rate;
|
|
|
|
bus_->tia_->set_output_mode(TIA::OutputMode::PAL);
|
|
|
|
}
|
|
|
|
|
2017-12-17 21:26:06 -05:00
|
|
|
bus_->speaker_.set_input_rate(static_cast<float>(clock_rate / static_cast<double>(CPUTicksPerAudioTick)));
|
|
|
|
bus_->speaker_.set_high_frequency_cutoff(static_cast<float>(clock_rate / (static_cast<double>(CPUTicksPerAudioTick) * 2.0)));
|
2017-08-16 14:52:40 -04:00
|
|
|
set_clock_rate(clock_rate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-07 14:26:07 -05:00
|
|
|
float get_confidence() override {
|
|
|
|
return confidence_counter_.get_confidence();
|
|
|
|
}
|
|
|
|
|
2017-08-16 14:52:40 -04:00
|
|
|
private:
|
|
|
|
// the bus
|
|
|
|
std::unique_ptr<Bus> bus_;
|
|
|
|
|
|
|
|
// 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];
|
2018-03-07 14:26:07 -05:00
|
|
|
unsigned int frame_record_pointer_ = 0;
|
|
|
|
bool is_ntsc_ = true;
|
2017-10-15 20:49:47 -04:00
|
|
|
std::vector<std::unique_ptr<Inputs::Joystick>> joysticks_;
|
2018-03-07 14:26:07 -05:00
|
|
|
|
|
|
|
// a confidence counter
|
|
|
|
Analyser::Dynamic::ConfidenceCounter confidence_counter_;
|
2017-08-16 14:52:40 -04:00
|
|
|
};
|
|
|
|
|
2017-02-19 21:20:37 -05:00
|
|
|
}
|
2017-08-16 14:52:40 -04:00
|
|
|
|
|
|
|
using namespace Atari2600;
|
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
Machine *Machine::Atari2600(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher) {
|
|
|
|
using Target = Analyser::Static::Atari::Target;
|
|
|
|
const Target *const atari_target = dynamic_cast<const Target *>(target);
|
|
|
|
return new Atari2600::ConcreteMachine(*atari_target);
|
2017-08-16 14:52:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
Machine::~Machine() {}
|