2016-10-11 22:20:13 -04:00
|
|
|
//
|
|
|
|
// Oric.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 11/10/2016.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2016 Thomas Harte. All rights reserved.
|
2016-10-11 22:20:13 -04:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "Oric.hpp"
|
2017-08-03 11:42:31 -04:00
|
|
|
|
2020-01-14 23:15:27 -05:00
|
|
|
#include "BD500.hpp"
|
2020-01-05 20:05:55 -05:00
|
|
|
#include "Jasmin.hpp"
|
2017-10-21 10:52:35 -04:00
|
|
|
#include "Keyboard.hpp"
|
2017-10-12 22:25:02 -04:00
|
|
|
#include "Microdisc.hpp"
|
|
|
|
#include "Video.hpp"
|
2017-08-16 14:35:53 -04:00
|
|
|
|
2018-05-11 23:05:36 -04:00
|
|
|
#include "../../Activity/Source.hpp"
|
2020-04-01 23:19:34 -04:00
|
|
|
#include "../MachineTypes.hpp"
|
2018-03-09 15:19:02 -05:00
|
|
|
|
2017-10-21 10:30:02 -04:00
|
|
|
#include "../Utility/MemoryFuzzer.hpp"
|
2018-05-13 13:57:19 -04:00
|
|
|
#include "../Utility/StringSerialiser.hpp"
|
2016-10-11 22:20:13 -04:00
|
|
|
|
2020-10-18 21:43:08 -04:00
|
|
|
#include "../../Processors/6502Esque/6502Selector.hpp"
|
2017-08-16 14:35:53 -04:00
|
|
|
#include "../../Components/6522/6522.hpp"
|
|
|
|
#include "../../Components/AY38910/AY38910.hpp"
|
2018-05-08 22:05:43 -04:00
|
|
|
#include "../../Components/DiskII/DiskII.hpp"
|
2017-08-03 11:42:31 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
#include "../../Storage/Tape/Tape.hpp"
|
|
|
|
#include "../../Storage/Tape/Parsers/Oric.hpp"
|
2016-10-11 22:20:13 -04:00
|
|
|
|
2017-08-21 21:56:42 -04:00
|
|
|
#include "../../ClockReceiver/ForceInline.hpp"
|
2017-11-20 21:55:32 -05:00
|
|
|
#include "../../Configurable/StandardOptions.hpp"
|
2017-12-18 21:39:23 -05:00
|
|
|
#include "../../Outputs/Speaker/Implementation/LowpassSpeaker.hpp"
|
2017-08-21 21:56:42 -04:00
|
|
|
|
2018-03-09 16:07:29 -05:00
|
|
|
#include "../../Analyser/Static/Oric/Target.hpp"
|
|
|
|
|
2021-04-04 20:43:16 -04:00
|
|
|
#include "../../ClockReceiver/JustInTime.hpp"
|
|
|
|
|
2017-11-24 16:59:00 -05:00
|
|
|
#include <cstdint>
|
2017-08-16 14:35:53 -04:00
|
|
|
#include <memory>
|
2017-11-24 16:59:00 -05:00
|
|
|
#include <vector>
|
2017-08-16 14:35:53 -04:00
|
|
|
|
2021-04-29 18:29:29 -04:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Provides an Altai-style joystick.
|
|
|
|
*/
|
|
|
|
class Joystick: public Inputs::ConcreteJoystick {
|
|
|
|
public:
|
|
|
|
Joystick() :
|
|
|
|
ConcreteJoystick({
|
|
|
|
Input(Input::Up),
|
|
|
|
Input(Input::Down),
|
|
|
|
Input(Input::Left),
|
|
|
|
Input(Input::Right),
|
|
|
|
Input(Input::Fire)
|
|
|
|
}) {}
|
|
|
|
|
|
|
|
void did_set_input(const Input &digital_input, bool is_active) final {
|
|
|
|
#define APPLY(b) if(is_active) state_ &= ~b; else state_ |= b;
|
|
|
|
switch(digital_input.type) {
|
|
|
|
default: return;
|
|
|
|
case Input::Right: APPLY(0x02); break;
|
|
|
|
case Input::Left: APPLY(0x01); break;
|
|
|
|
case Input::Down: APPLY(0x08); break;
|
|
|
|
case Input::Up: APPLY(0x10); break;
|
|
|
|
case Input::Fire: APPLY(0x20); break;
|
|
|
|
}
|
|
|
|
#undef APPLY
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t get_state() {
|
|
|
|
return state_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint8_t state_ = 0xff;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
namespace Oric {
|
|
|
|
|
2020-01-05 20:34:15 -05:00
|
|
|
using DiskInterface = Analyser::Static::Oric::Target::DiskInterface;
|
2020-10-18 21:43:08 -04:00
|
|
|
using Processor = Analyser::Static::Oric::Target::Processor;
|
2020-02-16 18:28:03 -05:00
|
|
|
using AY = GI::AY38910::AY38910<false>;
|
2021-11-21 15:37:29 -05:00
|
|
|
using Speaker = Outputs::Speaker::PullLowpass<AY>;
|
2020-01-05 20:34:15 -05:00
|
|
|
|
2017-11-24 16:59:00 -05:00
|
|
|
enum ROM {
|
|
|
|
BASIC10 = 0, BASIC11, Microdisc, Colour
|
|
|
|
};
|
|
|
|
|
2017-09-04 18:22:14 -04:00
|
|
|
/*!
|
|
|
|
Models the Oric's keyboard: eight key rows, containing a bitfield of keys set.
|
|
|
|
|
|
|
|
Active line is selected through a port on the Oric's VIA, and a column mask is
|
|
|
|
selected via a port on the AY, returning a single Boolean representation of the
|
|
|
|
logical OR of every key selected by the column mask on the active row.
|
|
|
|
*/
|
|
|
|
class Keyboard {
|
|
|
|
public:
|
2020-01-05 21:57:33 -05:00
|
|
|
struct SpecialKeyHandler {
|
|
|
|
virtual void perform_special_key(Oric::Key key) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
Keyboard(SpecialKeyHandler *handler) : special_key_handler_(handler) {
|
2017-09-04 18:22:14 -04:00
|
|
|
clear_all_keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets whether @c key is or is not pressed, per @c is_pressed.
|
|
|
|
void set_key_state(uint16_t key, bool is_pressed) {
|
2020-01-05 21:57:33 -05:00
|
|
|
switch(key) {
|
|
|
|
default: {
|
|
|
|
const uint8_t mask = key & 0xff;
|
|
|
|
const int line = key >> 8;
|
|
|
|
|
|
|
|
if(is_pressed) rows_[line] |= mask;
|
|
|
|
else rows_[line] &= ~mask;
|
|
|
|
} break;
|
2017-09-04 18:22:14 -04:00
|
|
|
|
2020-01-05 21:57:33 -05:00
|
|
|
case KeyNMI:
|
|
|
|
case KeyJasminReset:
|
|
|
|
if(is_pressed) {
|
|
|
|
special_key_handler_->perform_special_key(Oric::Key(key));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-09-04 18:22:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets all keys as unpressed.
|
|
|
|
void clear_all_keys() {
|
|
|
|
memset(rows_, 0, sizeof(rows_));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Selects the active row.
|
|
|
|
void set_active_row(uint8_t row) {
|
|
|
|
row_ = row & 7;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Queries the keys on the active row specified by @c mask.
|
|
|
|
bool query_column(uint8_t column_mask) {
|
|
|
|
return !!(rows_[row_] & column_mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint8_t row_ = 0;
|
|
|
|
uint8_t rows_[8];
|
2020-01-05 21:57:33 -05:00
|
|
|
SpecialKeyHandler *const special_key_handler_;
|
2017-09-04 18:22:14 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Provide's the Oric's tape player: a standard binary-sampled tape which also holds
|
|
|
|
an instance of the Oric tape parser, to provide fast-tape loading.
|
|
|
|
*/
|
|
|
|
class TapePlayer: public Storage::Tape::BinaryTapePlayer {
|
|
|
|
public:
|
|
|
|
TapePlayer() : Storage::Tape::BinaryTapePlayer(1000000) {}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Parses the incoming tape event stream to obtain the next stored byte.
|
|
|
|
|
|
|
|
@param use_fast_encoding If set to @c true , inspects the tape as though it
|
|
|
|
is encoded in the Oric's fast-loading scheme. Otherwise looks for a slow-encoded byte.
|
|
|
|
|
|
|
|
@returns The next byte from the tape.
|
|
|
|
*/
|
|
|
|
uint8_t get_next_byte(bool use_fast_encoding) {
|
2020-05-09 23:00:39 -04:00
|
|
|
return uint8_t(parser_.get_next_byte(get_tape(), use_fast_encoding));
|
2017-09-04 18:22:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Storage::Tape::Oric::Parser parser_;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Implements the Oric's VIA's port handler. On the Oric the VIA's ports connect
|
|
|
|
to the AY, the tape's motor control signal and the keyboard.
|
|
|
|
*/
|
|
|
|
class VIAPortHandler: public MOS::MOS6522::IRQDelegatePortHandler {
|
|
|
|
public:
|
2022-07-16 14:41:04 -04:00
|
|
|
VIAPortHandler(Concurrency::AsyncTaskQueue<false> &audio_queue, AY &ay8910, Speaker &speaker, TapePlayer &tape_player, Keyboard &keyboard) :
|
2021-04-29 18:29:29 -04:00
|
|
|
audio_queue_(audio_queue), ay8910_(ay8910), speaker_(speaker), tape_player_(tape_player), keyboard_(keyboard)
|
|
|
|
{
|
|
|
|
// Attach a couple of joysticks.
|
|
|
|
joysticks_.emplace_back(new Joystick);
|
|
|
|
joysticks_.emplace_back(new Joystick);
|
|
|
|
}
|
2017-09-04 18:22:14 -04:00
|
|
|
|
|
|
|
/*!
|
|
|
|
Reponds to the 6522's control line output change signal; on an Oric A2 is connected to
|
|
|
|
the AY's BDIR, and B2 is connected to the AY's A2.
|
|
|
|
*/
|
|
|
|
void set_control_line_output(MOS::MOS6522::Port port, MOS::MOS6522::Line line, bool value) {
|
|
|
|
if(line) {
|
|
|
|
if(port) ay_bdir_ = value; else ay_bc1_ = value;
|
|
|
|
update_ay();
|
2017-12-17 21:26:06 -05:00
|
|
|
ay8910_.set_control_lines( (GI::AY38910::ControlLines)((ay_bdir_ ? GI::AY38910::BDIR : 0) | (ay_bc1_ ? GI::AY38910::BC1 : 0) | GI::AY38910::BC2));
|
2017-09-04 18:22:14 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Reponds to changes in the 6522's port output. On an Oric port B sets the tape motor control
|
|
|
|
and the keyboard's active row. Port A is connected to the AY's data bus.
|
|
|
|
*/
|
2023-05-16 16:40:09 -04:00
|
|
|
void set_port_output(MOS::MOS6522::Port port, uint8_t value, uint8_t) {
|
2017-09-04 18:22:14 -04:00
|
|
|
if(port) {
|
|
|
|
keyboard_.set_active_row(value);
|
|
|
|
tape_player_.set_motor_control(value & 0x40);
|
|
|
|
} else {
|
|
|
|
update_ay();
|
2017-12-17 21:26:06 -05:00
|
|
|
ay8910_.set_data_input(value);
|
2021-04-29 18:29:29 -04:00
|
|
|
porta_output_ = value;
|
2017-09-04 18:22:14 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Provides input data for the 6522. Port B reads the keyboard, and port A reads from the AY.
|
|
|
|
*/
|
|
|
|
uint8_t get_port_input(MOS::MOS6522::Port port) {
|
|
|
|
if(port) {
|
2017-12-17 21:26:06 -05:00
|
|
|
uint8_t column = ay8910_.get_port_output(false) ^ 0xff;
|
2017-09-04 18:22:14 -04:00
|
|
|
return keyboard_.query_column(column) ? 0x08 : 0x00;
|
|
|
|
} else {
|
2021-04-29 18:29:29 -04:00
|
|
|
uint8_t result = ay8910_.get_data_output();
|
|
|
|
if(porta_output_ & 0x40) result &= static_cast<Joystick *>(joysticks_[0].get())->get_state();
|
|
|
|
if(porta_output_ & 0x80) result &= static_cast<Joystick *>(joysticks_[1].get())->get_state();
|
|
|
|
return result;
|
2017-09-04 18:22:14 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Advances time. This class manages the AY's concept of time to permit updating-on-demand.
|
|
|
|
*/
|
2019-06-01 14:39:40 -04:00
|
|
|
inline void run_for(const HalfCycles cycles) {
|
2017-09-04 18:22:14 -04:00
|
|
|
cycles_since_ay_update_ += cycles;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Flushes any queued behaviour (which, specifically, means on the AY).
|
|
|
|
void flush() {
|
2017-12-17 21:26:06 -05:00
|
|
|
update_ay();
|
|
|
|
audio_queue_.perform();
|
2017-09-04 18:22:14 -04:00
|
|
|
}
|
|
|
|
|
2021-04-29 18:29:29 -04:00
|
|
|
const std::vector<std::unique_ptr<Inputs::Joystick>> &get_joysticks() {
|
|
|
|
return joysticks_;
|
|
|
|
}
|
|
|
|
|
2017-09-04 18:22:14 -04:00
|
|
|
private:
|
|
|
|
void update_ay() {
|
2019-07-28 21:49:54 -04:00
|
|
|
speaker_.run_for(audio_queue_, cycles_since_ay_update_.flush<Cycles>());
|
2017-09-04 18:22:14 -04:00
|
|
|
}
|
|
|
|
bool ay_bdir_ = false;
|
|
|
|
bool ay_bc1_ = false;
|
2021-04-29 18:29:29 -04:00
|
|
|
uint8_t porta_output_ = 0xff;
|
2019-06-01 14:39:40 -04:00
|
|
|
HalfCycles cycles_since_ay_update_;
|
2017-09-04 18:22:14 -04:00
|
|
|
|
2022-07-16 14:41:04 -04:00
|
|
|
Concurrency::AsyncTaskQueue<false> &audio_queue_;
|
2020-02-16 18:28:03 -05:00
|
|
|
AY &ay8910_;
|
2020-02-15 18:03:12 -05:00
|
|
|
Speaker &speaker_;
|
2017-09-04 18:22:14 -04:00
|
|
|
TapePlayer &tape_player_;
|
|
|
|
Keyboard &keyboard_;
|
2021-04-29 18:29:29 -04:00
|
|
|
|
|
|
|
std::vector<std::unique_ptr<Inputs::Joystick>> joysticks_;
|
2017-09-04 18:22:14 -04:00
|
|
|
};
|
|
|
|
|
2020-10-18 21:43:08 -04:00
|
|
|
template <Analyser::Static::Oric::Target::DiskInterface disk_interface, CPU::MOS6502Esque::Type processor_type> class ConcreteMachine:
|
2020-04-01 23:19:34 -04:00
|
|
|
public MachineTypes::TimedMachine,
|
|
|
|
public MachineTypes::ScanProducer,
|
|
|
|
public MachineTypes::AudioProducer,
|
2021-04-29 18:29:29 -04:00
|
|
|
public MachineTypes::JoystickMachine,
|
2020-04-01 23:19:34 -04:00
|
|
|
public MachineTypes::MediaTarget,
|
|
|
|
public MachineTypes::MappedKeyboardMachine,
|
2018-03-09 15:19:02 -05:00
|
|
|
public Configurable::Device,
|
2017-08-16 14:35:53 -04:00
|
|
|
public CPU::MOS6502::BusHandler,
|
2017-09-04 14:26:04 -04:00
|
|
|
public MOS::MOS6522::IRQDelegatePortHandler::Delegate,
|
2017-08-16 14:35:53 -04:00
|
|
|
public Storage::Tape::BinaryTapePlayer::Delegate,
|
2020-01-14 22:23:00 -05:00
|
|
|
public DiskController::Delegate,
|
2018-05-11 23:05:36 -04:00
|
|
|
public Activity::Source,
|
2020-01-05 21:57:33 -05:00
|
|
|
public Machine,
|
|
|
|
public Keyboard::SpecialKeyHandler {
|
2017-08-16 14:35:53 -04:00
|
|
|
|
|
|
|
public:
|
2018-07-10 20:00:46 -04:00
|
|
|
ConcreteMachine(const Analyser::Static::Oric::Target &target, const ROMMachine::ROMFetcher &rom_fetcher) :
|
2018-08-13 22:17:22 -04:00
|
|
|
m6502_(*this),
|
2021-04-04 20:43:16 -04:00
|
|
|
video_(ram_),
|
2019-12-18 19:28:41 -05:00
|
|
|
ay8910_(GI::AY38910::Personality::AY38910, audio_queue_),
|
2017-12-17 21:26:06 -05:00
|
|
|
speaker_(ay8910_),
|
|
|
|
via_port_handler_(audio_queue_, ay8910_, speaker_, tape_player_, keyboard_),
|
2018-05-28 17:19:29 -04:00
|
|
|
via_(via_port_handler_),
|
2020-01-05 21:57:33 -05:00
|
|
|
keyboard_(this),
|
2018-05-28 17:19:29 -04:00
|
|
|
diskii_(2000000) {
|
2017-08-16 14:35:53 -04:00
|
|
|
set_clock_rate(1000000);
|
2019-02-05 21:42:39 -05:00
|
|
|
speaker_.set_input_rate(1000000.0f);
|
2017-09-04 14:26:04 -04:00
|
|
|
via_port_handler_.set_interrupt_delegate(this);
|
2017-09-04 18:22:14 -04:00
|
|
|
tape_player_.set_delegate(this);
|
2020-01-15 23:47:45 -05:00
|
|
|
|
|
|
|
// Slight hack here: I'm unclear what RAM should look like at startup.
|
|
|
|
// Actually, I think completely random might be right since the Microdisc
|
|
|
|
// sort of assumes it, but also the BD-500 never explicitly sets PAL mode
|
|
|
|
// so I can't have any switch-to-NTSC bytes in the display area. Hence:
|
|
|
|
// disallow all atributes.
|
2017-08-16 14:35:53 -04:00
|
|
|
Memory::Fuzz(ram_, sizeof(ram_));
|
2020-01-15 23:47:45 -05:00
|
|
|
for(size_t c = 0; c < sizeof(ram_); ++c) {
|
2020-01-16 00:01:16 -05:00
|
|
|
ram_[c] |= 0x40;
|
2020-01-15 23:47:45 -05:00
|
|
|
}
|
2018-05-19 23:15:28 -04:00
|
|
|
|
2021-06-03 21:55:59 -04:00
|
|
|
::ROM::Request request = ::ROM::Request(::ROM::Name::OricColourROM, true);
|
|
|
|
::ROM::Name basic;
|
2018-07-10 20:00:46 -04:00
|
|
|
switch(target.rom) {
|
2021-06-03 22:46:47 -04:00
|
|
|
case Analyser::Static::Oric::Target::ROM::BASIC10: basic = ::ROM::Name::OricBASIC10; break;
|
|
|
|
default:
|
|
|
|
case Analyser::Static::Oric::Target::ROM::BASIC11: basic = ::ROM::Name::OricBASIC11; break;
|
2021-06-03 21:55:59 -04:00
|
|
|
case Analyser::Static::Oric::Target::ROM::Pravetz: basic = ::ROM::Name::OricPravetzBASIC; break;
|
2018-05-08 22:05:43 -04:00
|
|
|
}
|
2021-06-03 21:55:59 -04:00
|
|
|
request = request && ::ROM::Request(basic);
|
|
|
|
|
2018-05-08 22:05:43 -04:00
|
|
|
switch(disk_interface) {
|
|
|
|
default: break;
|
2020-01-14 22:23:00 -05:00
|
|
|
case DiskInterface::BD500:
|
2021-06-03 21:55:59 -04:00
|
|
|
request = request && ::ROM::Request(::ROM::Name::OricByteDrive500);
|
2020-01-14 22:23:00 -05:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Jasmin:
|
2021-06-03 21:55:59 -04:00
|
|
|
request = request && ::ROM::Request(::ROM::Name::OricJasmin);
|
2020-01-05 20:05:55 -05:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Microdisc:
|
2021-06-03 21:55:59 -04:00
|
|
|
request = request && ::ROM::Request(::ROM::Name::OricMicrodisc);
|
2019-07-22 22:15:44 -04:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Pravetz:
|
2021-06-03 21:55:59 -04:00
|
|
|
request = request && ::ROM::Request(::ROM::Name::Oric8DOSBoot) && ::ROM::Request(::ROM::Name::DiskIIStateMachine16Sector);
|
2019-07-22 22:15:44 -04:00
|
|
|
break;
|
2018-05-08 22:05:43 -04:00
|
|
|
}
|
|
|
|
|
2021-06-04 18:54:50 -04:00
|
|
|
auto roms = rom_fetcher(request);
|
2021-06-03 21:55:59 -04:00
|
|
|
if(!request.validate(roms)) {
|
|
|
|
throw ROMMachine::Error::MissingROMs;
|
2017-11-06 22:14:15 -05:00
|
|
|
}
|
2018-07-10 20:00:46 -04:00
|
|
|
|
2021-06-03 21:55:59 -04:00
|
|
|
// The colour ROM is optional; an alternative composite encoding can be used if
|
|
|
|
// it is absent.
|
|
|
|
const auto colour_rom = roms.find(::ROM::Name::OricColourROM);
|
|
|
|
if(colour_rom != roms.end()) {
|
|
|
|
video_->set_colour_rom(colour_rom->second);
|
|
|
|
}
|
|
|
|
rom_ = std::move(roms.find(basic)->second);
|
2018-05-08 22:05:43 -04:00
|
|
|
|
|
|
|
switch(disk_interface) {
|
|
|
|
default: break;
|
2020-01-14 22:23:00 -05:00
|
|
|
case DiskInterface::BD500:
|
2021-06-03 21:55:59 -04:00
|
|
|
disk_rom_ = std::move(roms.find(::ROM::Name::OricByteDrive500)->second);
|
2020-01-14 22:23:00 -05:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Jasmin:
|
2021-06-03 21:55:59 -04:00
|
|
|
disk_rom_ = std::move(roms.find(::ROM::Name::OricJasmin)->second);
|
2020-01-05 20:05:55 -05:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Microdisc:
|
2021-06-03 21:55:59 -04:00
|
|
|
disk_rom_ = std::move(roms.find(::ROM::Name::OricMicrodisc)->second);
|
2018-05-08 22:05:43 -04:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Pravetz: {
|
2021-06-03 21:55:59 -04:00
|
|
|
pravetz_rom_ = std::move(roms.find(::ROM::Name::Oric8DOSBoot)->second);
|
2018-05-08 22:05:43 -04:00
|
|
|
pravetz_rom_.resize(512);
|
|
|
|
|
2021-06-03 21:55:59 -04:00
|
|
|
diskii_->set_state_machine(roms.find(::ROM::Name::DiskIIStateMachine16Sector)->second);
|
2018-05-08 22:05:43 -04:00
|
|
|
} break;
|
|
|
|
}
|
|
|
|
|
|
|
|
paged_rom_ = rom_.data();
|
2017-11-24 22:22:32 -05:00
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
switch(target.disk_interface) {
|
2018-05-08 22:05:43 -04:00
|
|
|
default: break;
|
2020-01-14 22:53:27 -05:00
|
|
|
case DiskInterface::BD500:
|
2020-01-14 23:15:27 -05:00
|
|
|
bd500_.set_delegate(this);
|
2018-05-08 22:05:43 -04:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Jasmin:
|
2020-01-05 20:05:55 -05:00
|
|
|
jasmin_.set_delegate(this);
|
|
|
|
break;
|
2020-01-14 22:53:27 -05:00
|
|
|
case DiskInterface::Microdisc:
|
|
|
|
microdisc_.set_delegate(this);
|
|
|
|
break;
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-11-03 22:14:40 -04:00
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
if(!target.loading_command.empty()) {
|
|
|
|
type_string(target.loading_command);
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-10-11 22:20:13 -04:00
|
|
|
|
2020-01-05 21:57:33 -05:00
|
|
|
if(target.should_start_jasmin) {
|
|
|
|
// If Jasmin autostart is requested then plan to do so in 3 seconds; empirically long enough
|
|
|
|
// for the Oric to boot normally, before the Jasmin intercedes.
|
|
|
|
jasmin_reset_counter_ = 3000000;
|
|
|
|
}
|
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
switch(target.rom) {
|
2018-05-08 22:05:43 -04:00
|
|
|
case Analyser::Static::Oric::Target::ROM::BASIC10:
|
|
|
|
tape_get_byte_address_ = 0xe630;
|
|
|
|
tape_speed_address_ = 0x67;
|
|
|
|
break;
|
|
|
|
case Analyser::Static::Oric::Target::ROM::BASIC11:
|
|
|
|
case Analyser::Static::Oric::Target::ROM::Pravetz:
|
|
|
|
tape_get_byte_address_ = 0xe6c9;
|
|
|
|
tape_speed_address_ = 0x024d;
|
|
|
|
break;
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2017-08-17 10:48:29 -04:00
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
insert_media(target.media);
|
|
|
|
}
|
|
|
|
|
|
|
|
~ConcreteMachine() {
|
|
|
|
audio_queue_.flush();
|
|
|
|
}
|
|
|
|
|
2020-01-14 22:23:00 -05:00
|
|
|
void set_key_state(uint16_t key, bool is_pressed) final {
|
2018-07-10 20:00:46 -04:00
|
|
|
if(key == KeyNMI) {
|
|
|
|
m6502_.set_nmi_line(is_pressed);
|
|
|
|
} else {
|
|
|
|
keyboard_.set_key_state(key, is_pressed);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-14 22:23:00 -05:00
|
|
|
void clear_all_keys() final {
|
2018-07-10 20:00:46 -04:00
|
|
|
keyboard_.clear_all_keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_use_fast_tape_hack(bool activate) {
|
|
|
|
use_fast_tape_hack_ = activate;
|
2017-08-17 10:48:29 -04:00
|
|
|
}
|
|
|
|
|
2020-01-05 20:05:55 -05:00
|
|
|
template <typename DiskInterface> bool insert_disks(const Analyser::Static::Media &media, DiskInterface &interface, int num_drives) {
|
|
|
|
int drive_index = 0;
|
|
|
|
for(auto &disk : media.disks) {
|
|
|
|
interface.set_disk(disk, drive_index);
|
|
|
|
++drive_index;
|
|
|
|
if(drive_index == num_drives) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-01-14 22:23:00 -05:00
|
|
|
bool insert_media(const Analyser::Static::Media &media) final {
|
2018-05-08 22:05:43 -04:00
|
|
|
bool inserted = false;
|
|
|
|
|
2018-07-10 20:00:46 -04:00
|
|
|
if(!media.tapes.empty()) {
|
2017-09-04 18:22:14 -04:00
|
|
|
tape_player_.set_tape(media.tapes.front());
|
2018-05-08 22:05:43 -04:00
|
|
|
inserted = true;
|
2017-08-17 10:48:29 -04:00
|
|
|
}
|
|
|
|
|
2018-05-08 22:05:43 -04:00
|
|
|
if(!media.disks.empty()) {
|
|
|
|
switch(disk_interface) {
|
2020-01-14 23:15:27 -05:00
|
|
|
case DiskInterface::BD500: inserted |= insert_disks(media, bd500_, 4); break;
|
|
|
|
case DiskInterface::Jasmin: inserted |= insert_disks(media, jasmin_, 4); break;
|
|
|
|
case DiskInterface::Microdisc: inserted |= insert_disks(media, microdisc_, 4); break;
|
2021-04-04 20:43:16 -04:00
|
|
|
case DiskInterface::Pravetz: inserted |= insert_disks(media, *diskii_.last_valid(), 2); break;
|
2018-05-08 22:05:43 -04:00
|
|
|
default: break;
|
|
|
|
}
|
2017-08-17 10:48:29 -04:00
|
|
|
}
|
|
|
|
|
2018-05-08 22:05:43 -04:00
|
|
|
return inserted;
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-10-11 22:20:13 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// to satisfy CPU::MOS6502::BusHandler
|
2017-08-21 21:56:42 -04:00
|
|
|
forceinline Cycles perform_bus_operation(CPU::MOS6502::BusOperation operation, uint16_t address, uint8_t *value) {
|
2017-08-16 14:35:53 -04:00
|
|
|
if(address > ram_top_) {
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = paged_rom_[address - ram_top_ - 1];
|
2017-08-16 14:35:53 -04:00
|
|
|
|
|
|
|
// 024D = 0 => fast; otherwise slow
|
|
|
|
// E6C9 = read byte: return byte in A
|
2017-09-04 18:22:14 -04:00
|
|
|
if( address == tape_get_byte_address_ &&
|
2018-05-08 22:05:43 -04:00
|
|
|
paged_rom_ == rom_.data() &&
|
2017-09-04 18:22:14 -04:00
|
|
|
use_fast_tape_hack_ &&
|
|
|
|
operation == CPU::MOS6502::BusOperation::ReadOpcode &&
|
|
|
|
tape_player_.has_tape() &&
|
|
|
|
!tape_player_.get_tape()->is_at_end()) {
|
|
|
|
|
|
|
|
uint8_t next_byte = tape_player_.get_next_byte(!ram_[tape_speed_address_]);
|
2023-05-10 18:53:38 -05:00
|
|
|
m6502_.set_value_of(CPU::MOS6502Esque::A, next_byte);
|
|
|
|
m6502_.set_value_of(CPU::MOS6502Esque::Flags, next_byte ? 0 : CPU::MOS6502::Flag::Zero);
|
2017-08-16 14:35:53 -04:00
|
|
|
*value = 0x60; // i.e. RTS
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if((address & 0xff00) == 0x0300) {
|
2020-01-05 20:34:15 -05:00
|
|
|
if(address < 0x0310 || (disk_interface == DiskInterface::None)) {
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = via_.read(address);
|
2020-01-05 13:40:02 -05:00
|
|
|
else via_.write(address, *value);
|
2018-05-08 22:05:43 -04:00
|
|
|
} else {
|
|
|
|
switch(disk_interface) {
|
|
|
|
default: break;
|
2020-01-14 23:15:27 -05:00
|
|
|
case DiskInterface::BD500:
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = bd500_.read(address);
|
2020-01-14 23:15:27 -05:00
|
|
|
else bd500_.write(address, *value);
|
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Jasmin:
|
2020-01-05 20:05:55 -05:00
|
|
|
if(address >= 0x3f4) {
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = jasmin_.read(address);
|
2020-01-05 20:05:55 -05:00
|
|
|
else jasmin_.write(address, *value);
|
|
|
|
}
|
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Microdisc:
|
2018-05-08 22:05:43 -04:00
|
|
|
switch(address) {
|
|
|
|
case 0x0310: case 0x0311: case 0x0312: case 0x0313:
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = microdisc_.read(address);
|
2020-01-05 13:40:02 -05:00
|
|
|
else microdisc_.write(address, *value);
|
2018-05-08 22:05:43 -04:00
|
|
|
break;
|
|
|
|
case 0x314: case 0x315: case 0x316: case 0x317:
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = microdisc_.get_interrupt_request_register();
|
2018-05-08 22:05:43 -04:00
|
|
|
else microdisc_.set_control_register(*value);
|
|
|
|
break;
|
|
|
|
case 0x318: case 0x319: case 0x31a: case 0x31b:
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = microdisc_.get_data_request_register();
|
2018-05-08 22:05:43 -04:00
|
|
|
break;
|
|
|
|
}
|
2017-08-16 14:35:53 -04:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Pravetz:
|
2018-05-08 22:05:43 -04:00
|
|
|
if(address >= 0x0320) {
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation)) *value = pravetz_rom_[pravetz_rom_base_pointer_ + (address & 0xff)];
|
2018-05-08 22:05:43 -04:00
|
|
|
else {
|
|
|
|
switch(address) {
|
|
|
|
case 0x380: case 0x381: case 0x382: case 0x383:
|
2018-05-09 20:28:25 -04:00
|
|
|
ram_top_ = (address&1) ? basic_invisible_ram_top_ : basic_visible_ram_top_;
|
2018-05-08 22:05:43 -04:00
|
|
|
pravetz_rom_base_pointer_ = (address&2) ? 0x100 : 0x000;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2021-04-04 20:43:16 -04:00
|
|
|
const int disk_value = diskii_->read_address(address);
|
|
|
|
if(!isWriteOperation(operation) && disk_value != Apple::DiskII::DidNotLoad) *value = uint8_t(disk_value);
|
2018-05-08 22:05:43 -04:00
|
|
|
}
|
2017-08-16 14:35:53 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2021-02-02 20:52:34 -05:00
|
|
|
if(!isWriteOperation(operation))
|
2017-08-16 14:35:53 -04:00
|
|
|
*value = ram_[address];
|
|
|
|
else {
|
2021-04-04 20:43:16 -04:00
|
|
|
if(address >= 0x9800 && address <= 0xc000) video_.flush();
|
2017-08-16 14:35:53 -04:00
|
|
|
ram_[address] = *value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-10-13 20:50:55 -04:00
|
|
|
|
2018-05-13 15:34:31 -04:00
|
|
|
// $02df is where the Oric ROMs; all of them, including BASIC 1.0, 1.1 and the Pravetz; have the
|
2018-05-13 14:02:46 -04:00
|
|
|
// IRQ routine store an incoming keystroke in order for reading to occur later. By capturing the
|
|
|
|
// read rather than the decode and write: (i) nothing is lost while BASIC is parsing; and
|
|
|
|
// (ii) keyboard input is much more rapid.
|
|
|
|
if(string_serialiser_ && address == 0x02df && operation == CPU::MOS6502::BusOperation::Read) {
|
|
|
|
*value = string_serialiser_->head() | 0x80;
|
2018-05-13 13:57:19 -04:00
|
|
|
if(!string_serialiser_->advance()) string_serialiser_.reset();
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-10-14 21:18:03 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
via_.run_for(Cycles(1));
|
2017-09-04 18:22:14 -04:00
|
|
|
tape_player_.run_for(Cycles(1));
|
2018-05-08 22:05:43 -04:00
|
|
|
switch(disk_interface) {
|
|
|
|
default: break;
|
2020-01-14 23:15:27 -05:00
|
|
|
case DiskInterface::BD500:
|
2020-01-15 23:15:39 -05:00
|
|
|
bd500_.run_for(Cycles(9)); // i.e. effective clock rate of 9Mhz.
|
2020-01-14 23:15:27 -05:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Jasmin:
|
2021-04-04 20:43:16 -04:00
|
|
|
jasmin_.run_for(Cycles(8)); // i.e. effective clock rate of 8Mhz.
|
2020-01-05 20:34:15 -05:00
|
|
|
|
|
|
|
// Jasmin autostart hack: wait for a period, then trigger a reset, having forced
|
2020-01-05 21:35:20 -05:00
|
|
|
// the Jasmin to page its ROM in first. I assume the latter being what the Jasmin's
|
|
|
|
// hardware boot button did.
|
2020-01-05 20:34:15 -05:00
|
|
|
if(jasmin_reset_counter_) {
|
|
|
|
--jasmin_reset_counter_;
|
|
|
|
if(!jasmin_reset_counter_) {
|
2020-01-05 21:57:33 -05:00
|
|
|
perform_special_key(KeyJasminReset);
|
2020-01-05 20:34:15 -05:00
|
|
|
}
|
|
|
|
}
|
2020-01-05 20:05:55 -05:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Microdisc:
|
2021-04-04 20:43:16 -04:00
|
|
|
microdisc_.run_for(Cycles(8)); // i.e. effective clock rate of 8Mhz.
|
2018-05-17 21:39:11 -04:00
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Pravetz:
|
2021-04-04 20:43:16 -04:00
|
|
|
if(diskii_.clocking_preference() == ClockingHint::Preference::RealTime) {
|
|
|
|
diskii_->set_data_input(*value);
|
2018-05-19 23:15:28 -04:00
|
|
|
}
|
2021-04-04 20:43:16 -04:00
|
|
|
diskii_ += Cycles(2); // i.e. effective clock rate of 2Mhz.
|
2018-05-17 21:39:11 -04:00
|
|
|
break;
|
2018-05-08 22:05:43 -04:00
|
|
|
}
|
2020-01-05 20:34:15 -05:00
|
|
|
|
2021-04-04 20:43:16 -04:00
|
|
|
video_ += Cycles(1);
|
2017-08-16 14:35:53 -04:00
|
|
|
return Cycles(1);
|
|
|
|
}
|
2016-10-14 21:18:03 -04:00
|
|
|
|
2022-07-09 13:33:46 -04:00
|
|
|
void flush_output(int outputs) final {
|
|
|
|
if(outputs & Output::Video) {
|
2022-07-08 16:04:32 -04:00
|
|
|
video_.flush();
|
|
|
|
}
|
2022-07-09 13:33:46 -04:00
|
|
|
if(outputs & Output::Audio) {
|
2022-07-08 16:04:32 -04:00
|
|
|
via_.flush();
|
|
|
|
}
|
2021-04-04 20:43:16 -04:00
|
|
|
diskii_.flush();
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-10-15 21:21:18 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// to satisfy CRTMachine::Machine
|
2020-01-14 22:23:00 -05:00
|
|
|
void set_scan_target(Outputs::Display::ScanTarget *scan_target) final {
|
2021-04-04 20:43:16 -04:00
|
|
|
video_.last_valid()->set_scan_target(scan_target);
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-11-03 07:59:30 -04:00
|
|
|
|
2020-01-21 22:28:25 -05:00
|
|
|
Outputs::Display::ScanStatus get_scaled_scan_status() const final {
|
2021-04-04 20:43:16 -04:00
|
|
|
return video_.last_valid()->get_scaled_scan_status();
|
2020-01-20 21:45:10 -05:00
|
|
|
}
|
|
|
|
|
2020-01-14 22:23:00 -05:00
|
|
|
void set_display_type(Outputs::Display::DisplayType display_type) final {
|
2021-04-04 20:43:16 -04:00
|
|
|
video_.last_valid()->set_display_type(display_type);
|
2018-11-29 20:44:21 -08:00
|
|
|
}
|
|
|
|
|
2020-05-20 23:34:26 -04:00
|
|
|
Outputs::Display::DisplayType get_display_type() const final {
|
2021-04-04 20:43:16 -04:00
|
|
|
return video_.last_valid()->get_display_type();
|
2020-03-18 18:31:31 -04:00
|
|
|
}
|
|
|
|
|
2020-01-14 22:23:00 -05:00
|
|
|
Outputs::Speaker::Speaker *get_speaker() final {
|
2017-12-17 21:26:06 -05:00
|
|
|
return &speaker_;
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-10-30 22:51:08 -04:00
|
|
|
|
2020-01-14 22:23:00 -05:00
|
|
|
void run_for(const Cycles cycles) final {
|
2017-08-16 14:35:53 -04:00
|
|
|
m6502_.run_for(cycles);
|
|
|
|
}
|
2016-10-30 22:51:08 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// to satisfy MOS::MOS6522IRQDelegate::Delegate
|
2020-05-30 00:37:06 -04:00
|
|
|
void mos6522_did_change_interrupt_status(void *) final {
|
2017-08-16 14:35:53 -04:00
|
|
|
set_interrupt_line();
|
|
|
|
}
|
2016-10-30 22:51:08 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// to satisfy Storage::Tape::BinaryTapePlayer::Delegate
|
2020-01-14 22:23:00 -05:00
|
|
|
void tape_did_change_input(Storage::Tape::BinaryTapePlayer *tape_player) final {
|
2017-08-16 14:35:53 -04:00
|
|
|
// set CB1
|
2017-09-04 14:26:04 -04:00
|
|
|
via_.set_control_line_input(MOS::MOS6522::Port::B, MOS::MOS6522::Line::One, !tape_player->get_input());
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2016-10-20 20:53:17 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// for Utility::TypeRecipient::Delegate
|
2020-01-14 22:23:00 -05:00
|
|
|
void type_string(const std::string &string) final {
|
2019-12-23 21:31:46 -05:00
|
|
|
string_serialiser_ = std::make_unique<Utility::StringSerialiser>(string, true);
|
2017-08-16 14:35:53 -04:00
|
|
|
}
|
2017-03-26 14:34:47 -04:00
|
|
|
|
2020-05-20 23:34:26 -04:00
|
|
|
bool can_type(char c) const final {
|
2020-03-01 20:25:12 -05:00
|
|
|
// Make an effort to type the entire printable ASCII range.
|
|
|
|
return c >= 32 && c < 127;
|
|
|
|
}
|
|
|
|
|
2020-01-14 22:23:00 -05:00
|
|
|
// DiskController::Delegate
|
2020-01-14 22:53:27 -05:00
|
|
|
void disk_controller_did_change_paged_item(DiskController *controller) final {
|
|
|
|
switch(controller->get_paged_item()) {
|
2020-01-05 20:05:55 -05:00
|
|
|
default:
|
|
|
|
ram_top_ = basic_visible_ram_top_;
|
|
|
|
paged_rom_ = rom_.data();
|
|
|
|
break;
|
|
|
|
|
2020-01-14 22:53:27 -05:00
|
|
|
case DiskController::PagedItem::RAM:
|
2020-01-05 20:05:55 -05:00
|
|
|
ram_top_ = basic_invisible_ram_top_;
|
|
|
|
break;
|
|
|
|
|
2020-01-14 22:53:27 -05:00
|
|
|
case DiskController::PagedItem::DiskROM:
|
2020-01-14 22:23:00 -05:00
|
|
|
ram_top_ = uint16_t(0xffff - disk_rom_.size());
|
|
|
|
paged_rom_ = disk_rom_.data();
|
2020-01-05 20:05:55 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WD::WD1770::Delegate
|
2020-05-30 00:37:06 -04:00
|
|
|
void wd1770_did_change_output(WD::WD1770 *) final {
|
2017-08-16 14:35:53 -04:00
|
|
|
set_interrupt_line();
|
|
|
|
}
|
2016-10-20 20:53:17 -04:00
|
|
|
|
2020-01-23 22:57:51 -05:00
|
|
|
KeyboardMapper *get_keyboard_mapper() final {
|
2018-02-09 16:31:05 -05:00
|
|
|
return &keyboard_mapper_;
|
2017-10-12 22:25:02 -04:00
|
|
|
}
|
|
|
|
|
2017-11-18 19:48:10 -05:00
|
|
|
// MARK: - Configuration options.
|
2020-03-16 23:25:05 -04:00
|
|
|
std::unique_ptr<Reflection::Struct> get_options() final {
|
2020-03-18 18:31:31 -04:00
|
|
|
auto options = std::make_unique<Options>(Configurable::OptionsType::UserFriendly);
|
|
|
|
options->output = get_video_signal_configurable();
|
|
|
|
options->quickload = use_fast_tape_hack_;
|
|
|
|
return options;
|
2017-11-18 19:48:10 -05:00
|
|
|
}
|
|
|
|
|
2020-03-18 18:31:31 -04:00
|
|
|
void set_options(const std::unique_ptr<Reflection::Struct> &str) final {
|
|
|
|
const auto options = dynamic_cast<Options *>(str.get());
|
|
|
|
set_video_signal_configurable(options->output);
|
|
|
|
set_use_fast_tape_hack(options->quickload);
|
2017-11-18 19:48:10 -05:00
|
|
|
}
|
|
|
|
|
2020-01-23 22:57:51 -05:00
|
|
|
void set_activity_observer(Activity::Observer *observer) final {
|
2018-05-11 23:05:36 -04:00
|
|
|
switch(disk_interface) {
|
|
|
|
default: break;
|
2020-01-15 23:39:15 -05:00
|
|
|
case DiskInterface::BD500:
|
|
|
|
bd500_.set_activity_observer(observer);
|
|
|
|
break;
|
|
|
|
case DiskInterface::Jasmin:
|
|
|
|
jasmin_.set_activity_observer(observer);
|
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Microdisc:
|
2018-05-11 23:05:36 -04:00
|
|
|
microdisc_.set_activity_observer(observer);
|
|
|
|
break;
|
2020-01-05 20:34:15 -05:00
|
|
|
case DiskInterface::Pravetz:
|
2021-04-04 20:43:16 -04:00
|
|
|
diskii_->set_activity_observer(observer);
|
2018-05-11 23:05:36 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
private:
|
2018-05-09 20:28:25 -04:00
|
|
|
const uint16_t basic_invisible_ram_top_ = 0xffff;
|
|
|
|
const uint16_t basic_visible_ram_top_ = 0xbfff;
|
|
|
|
|
2020-10-18 21:43:08 -04:00
|
|
|
CPU::MOS6502Esque::Processor<processor_type, ConcreteMachine, false> m6502_;
|
2016-10-20 20:53:17 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// RAM and ROM
|
2020-01-14 22:23:00 -05:00
|
|
|
std::vector<uint8_t> rom_, disk_rom_;
|
2018-05-08 22:05:43 -04:00
|
|
|
uint8_t ram_[65536];
|
2016-10-20 20:53:17 -04:00
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// ROM bookkeeping
|
2018-05-13 14:02:46 -04:00
|
|
|
uint16_t tape_get_byte_address_ = 0, tape_speed_address_ = 0;
|
2017-09-04 18:22:14 -04:00
|
|
|
int keyboard_read_count_ = 0;
|
2017-08-16 14:35:53 -04:00
|
|
|
|
|
|
|
// Outputs
|
2021-04-04 20:43:16 -04:00
|
|
|
JustInTimeActor<VideoOutput, Cycles> video_;
|
2017-12-17 21:26:06 -05:00
|
|
|
|
2022-07-16 14:41:04 -04:00
|
|
|
Concurrency::AsyncTaskQueue<false> audio_queue_;
|
2020-02-16 18:28:03 -05:00
|
|
|
GI::AY38910::AY38910<false> ay8910_;
|
2020-02-15 18:03:12 -05:00
|
|
|
Speaker speaker_;
|
2017-08-16 14:35:53 -04:00
|
|
|
|
2017-10-12 22:25:02 -04:00
|
|
|
// Inputs
|
|
|
|
Oric::KeyboardMapper keyboard_mapper_;
|
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
// The tape
|
2017-09-04 18:22:14 -04:00
|
|
|
TapePlayer tape_player_;
|
|
|
|
bool use_fast_tape_hack_ = false;
|
2017-08-16 14:35:53 -04:00
|
|
|
|
2017-09-04 18:22:14 -04:00
|
|
|
VIAPortHandler via_port_handler_;
|
|
|
|
MOS::MOS6522::MOS6522<VIAPortHandler> via_;
|
|
|
|
Keyboard keyboard_;
|
2017-08-16 14:35:53 -04:00
|
|
|
|
2020-01-05 20:05:55 -05:00
|
|
|
// the Microdisc, if in use.
|
2017-08-16 14:35:53 -04:00
|
|
|
class Microdisc microdisc_;
|
2018-05-08 22:05:43 -04:00
|
|
|
|
2020-01-05 20:05:55 -05:00
|
|
|
// the Jasmin, if in use.
|
|
|
|
Jasmin jasmin_;
|
2020-01-05 21:57:33 -05:00
|
|
|
int jasmin_reset_counter_ = 0;
|
2020-01-05 20:05:55 -05:00
|
|
|
|
2020-01-14 23:15:27 -05:00
|
|
|
// the BD-500, if in use.
|
|
|
|
BD500 bd500_;
|
|
|
|
|
2020-01-05 20:05:55 -05:00
|
|
|
// the Pravetz/Disk II, if in use.
|
2021-04-04 20:43:16 -04:00
|
|
|
JustInTimeActor<Apple::DiskII, Cycles> diskii_;
|
2018-05-08 22:05:43 -04:00
|
|
|
std::vector<uint8_t> pravetz_rom_;
|
|
|
|
std::size_t pravetz_rom_base_pointer_ = 0;
|
|
|
|
|
|
|
|
// Overlay RAM
|
2018-05-09 20:28:25 -04:00
|
|
|
uint16_t ram_top_ = basic_visible_ram_top_;
|
2018-05-08 22:05:43 -04:00
|
|
|
uint8_t *paged_rom_ = nullptr;
|
2017-08-16 14:35:53 -04:00
|
|
|
|
2018-05-08 22:05:43 -04:00
|
|
|
// Helper to discern current IRQ state
|
2017-08-16 14:35:53 -04:00
|
|
|
inline void set_interrupt_line() {
|
2018-05-08 22:05:43 -04:00
|
|
|
bool irq_line = via_.get_interrupt_line();
|
2020-01-05 20:05:55 -05:00
|
|
|
|
|
|
|
// The Microdisc directly provides an interrupt line.
|
2020-01-05 20:34:15 -05:00
|
|
|
if constexpr (disk_interface == DiskInterface::Microdisc) {
|
2018-05-08 22:05:43 -04:00
|
|
|
irq_line |= microdisc_.get_interrupt_request_line();
|
2020-01-05 20:05:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// The Jasmin reroutes its data request line to the processor's interrupt line.
|
2020-01-05 20:34:15 -05:00
|
|
|
if constexpr (disk_interface == DiskInterface::Jasmin) {
|
2020-01-05 20:05:55 -05:00
|
|
|
irq_line |= jasmin_.get_data_request_line();
|
|
|
|
}
|
|
|
|
|
2018-05-08 22:05:43 -04:00
|
|
|
m6502_.set_irq_line(irq_line);
|
2016-11-25 20:15:48 +08:00
|
|
|
}
|
2018-05-13 13:57:19 -04:00
|
|
|
|
2020-01-05 21:57:33 -05:00
|
|
|
// Keys that aren't read by polling.
|
2020-01-23 22:57:51 -05:00
|
|
|
void perform_special_key(Oric::Key key) final {
|
2020-01-05 21:57:33 -05:00
|
|
|
switch(key) {
|
|
|
|
default: break;
|
|
|
|
|
|
|
|
case KeyJasminReset:
|
|
|
|
jasmin_.write(0x3fa, 0);
|
|
|
|
jasmin_.write(0x3fb, 1);
|
|
|
|
m6502_.set_power_on(true);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case KeyNMI:
|
|
|
|
// As luck would have it, the 6502's NMI line is edge triggered.
|
|
|
|
// So just forcing through an edge will work here.
|
|
|
|
m6502_.set_nmi_line(true);
|
|
|
|
m6502_.set_nmi_line(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-29 18:29:29 -04:00
|
|
|
// MARK: - typing
|
2018-05-13 13:57:19 -04:00
|
|
|
std::unique_ptr<Utility::StringSerialiser> string_serialiser_;
|
2021-04-29 18:29:29 -04:00
|
|
|
|
|
|
|
// MARK: - Joysticks
|
|
|
|
const std::vector<std::unique_ptr<Inputs::Joystick>> &get_joysticks() override {
|
|
|
|
return via_port_handler_.get_joysticks();
|
|
|
|
}
|
2017-08-16 14:35:53 -04:00
|
|
|
};
|
2016-11-25 20:15:48 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-08-16 14:35:53 -04:00
|
|
|
using namespace Oric;
|
|
|
|
|
2024-01-12 22:03:19 -05:00
|
|
|
std::unique_ptr<Machine> Machine::Oric(const Analyser::Static::Target *target_hint, const ROMMachine::ROMFetcher &rom_fetcher) {
|
2018-05-08 22:05:43 -04:00
|
|
|
auto *const oric_target = dynamic_cast<const Analyser::Static::Oric::Target *>(target_hint);
|
2020-10-18 21:43:08 -04:00
|
|
|
|
|
|
|
#define DiskInterfaceSwitch(processor) \
|
|
|
|
switch(oric_target->disk_interface) { \
|
2024-01-12 22:03:19 -05:00
|
|
|
default: return std::make_unique<ConcreteMachine<DiskInterface::None, processor>>(*oric_target, rom_fetcher); \
|
|
|
|
case DiskInterface::Microdisc: return std::make_unique<ConcreteMachine<DiskInterface::Microdisc, processor>>(*oric_target, rom_fetcher); \
|
|
|
|
case DiskInterface::Pravetz: return std::make_unique<ConcreteMachine<DiskInterface::Pravetz, processor>>(*oric_target, rom_fetcher); \
|
|
|
|
case DiskInterface::Jasmin: return std::make_unique<ConcreteMachine<DiskInterface::Jasmin, processor>>(*oric_target, rom_fetcher); \
|
|
|
|
case DiskInterface::BD500: return std::make_unique<ConcreteMachine<DiskInterface::BD500, processor>>(*oric_target, rom_fetcher); \
|
2020-10-18 21:43:08 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
switch(oric_target->processor) {
|
|
|
|
case Processor::WDC65816: DiskInterfaceSwitch(CPU::MOS6502Esque::Type::TWDC65816);
|
|
|
|
case Processor::MOS6502: DiskInterfaceSwitch(CPU::MOS6502Esque::Type::T6502);
|
2018-05-08 22:05:43 -04:00
|
|
|
}
|
2020-10-18 21:43:08 -04:00
|
|
|
|
|
|
|
#undef DiskInterfaceSwitch
|
|
|
|
|
2020-11-22 21:43:56 -05:00
|
|
|
return nullptr;
|
2016-11-25 20:15:48 +08:00
|
|
|
}
|