2016-09-25 21:24:16 -04:00
|
|
|
//
|
|
|
|
// Drive.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 25/09/2016.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2016 Thomas Harte. All rights reserved.
|
2016-09-25 21:24:16 -04:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "Drive.hpp"
|
2017-09-10 17:33:01 -04:00
|
|
|
|
2017-11-09 22:04:49 -05:00
|
|
|
#include "Track/UnformattedTrack.hpp"
|
2017-09-10 17:33:01 -04:00
|
|
|
|
2016-09-25 21:24:16 -04:00
|
|
|
#include <algorithm>
|
2017-09-10 17:33:01 -04:00
|
|
|
#include <cassert>
|
2018-05-14 20:01:20 -04:00
|
|
|
#include <cmath>
|
|
|
|
#include <chrono>
|
|
|
|
#include <random>
|
2016-09-25 21:24:16 -04:00
|
|
|
|
|
|
|
using namespace Storage::Disk;
|
|
|
|
|
2020-01-16 21:34:48 -05:00
|
|
|
Drive::Drive(int input_clock_rate, int revolutions_per_minute, int number_of_heads, ReadyType rdy_type):
|
2017-09-10 14:43:20 -04:00
|
|
|
Storage::TimedEventLoop(input_clock_rate),
|
2020-01-16 21:34:48 -05:00
|
|
|
available_heads_(number_of_heads),
|
|
|
|
ready_type_(rdy_type) {
|
2019-09-28 23:23:15 -04:00
|
|
|
set_rotation_speed(revolutions_per_minute);
|
2018-05-14 20:01:20 -04:00
|
|
|
|
2020-05-09 23:00:39 -04:00
|
|
|
const auto seed = std::default_random_engine::result_type(std::chrono::system_clock::now().time_since_epoch().count());
|
2018-05-14 20:01:20 -04:00
|
|
|
std::default_random_engine randomiser(seed);
|
|
|
|
|
|
|
|
// Get at least 64 bits of random information; rounding is likey to give this a slight bias.
|
|
|
|
random_source_ = 0;
|
|
|
|
auto half_range = (randomiser.max() - randomiser.min()) / 2;
|
|
|
|
for(int bit = 0; bit < 64; ++bit) {
|
|
|
|
random_source_ <<= 1;
|
|
|
|
random_source_ |= ((randomiser() - randomiser.min()) >= half_range) ? 1 : 0;
|
|
|
|
}
|
2017-09-10 14:43:20 -04:00
|
|
|
}
|
2016-09-25 21:24:16 -04:00
|
|
|
|
2020-01-16 21:34:48 -05:00
|
|
|
Drive::Drive(int input_clock_rate, int number_of_heads, ReadyType rdy_type) : Drive(input_clock_rate, 300, number_of_heads, rdy_type) {}
|
2019-06-04 21:41:54 -04:00
|
|
|
|
|
|
|
void Drive::set_rotation_speed(float revolutions_per_minute) {
|
2019-07-26 17:20:32 -04:00
|
|
|
// Rationalise the supplied speed so that cycles_per_revolution_ is exact.
|
|
|
|
cycles_per_revolution_ = int(0.5f + float(get_input_clock_rate()) * 60.0f / revolutions_per_minute);
|
|
|
|
|
|
|
|
// From there derive the appropriate rotational multiplier and possibly update the
|
|
|
|
// count of cycles since the index hole proportionally.
|
|
|
|
const float new_rotational_multiplier = float(cycles_per_revolution_) / float(get_input_clock_rate());
|
2020-03-29 18:36:57 -04:00
|
|
|
cycles_since_index_hole_ = Cycles::IntType(float(cycles_since_index_hole_) * new_rotational_multiplier / rotational_multiplier_);
|
2019-07-25 22:29:54 -04:00
|
|
|
rotational_multiplier_ = new_rotational_multiplier;
|
2019-07-26 17:20:32 -04:00
|
|
|
cycles_since_index_hole_ %= cycles_per_revolution_;
|
2019-06-04 21:41:54 -04:00
|
|
|
}
|
|
|
|
|
2017-10-07 19:14:18 -04:00
|
|
|
Drive::~Drive() {
|
|
|
|
if(disk_) disk_->flush_tracks();
|
|
|
|
}
|
|
|
|
|
2017-03-26 14:34:47 -04:00
|
|
|
void Drive::set_disk(const std::shared_ptr<Disk> &disk) {
|
2020-01-16 21:34:48 -05:00
|
|
|
if(ready_type_ == ReadyType::ShugartModifiedRDY || ready_type_ == ReadyType::IBMRDY) {
|
|
|
|
is_ready_ = false;
|
|
|
|
}
|
2020-11-18 17:20:48 -05:00
|
|
|
const bool had_disk = bool(disk_);
|
2017-10-07 19:14:18 -04:00
|
|
|
if(disk_) disk_->flush_tracks();
|
2016-12-03 11:59:28 -05:00
|
|
|
disk_ = disk;
|
2017-08-20 10:55:08 -04:00
|
|
|
has_disk_ = !!disk_;
|
2016-12-26 14:24:33 -05:00
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
invalidate_track();
|
2020-11-18 17:20:48 -05:00
|
|
|
did_set_disk(had_disk);
|
2018-05-27 23:17:06 -04:00
|
|
|
update_clocking_observer();
|
2016-09-25 21:24:16 -04:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::has_disk() const {
|
2017-08-20 10:55:08 -04:00
|
|
|
return has_disk_;
|
2016-09-25 21:24:16 -04:00
|
|
|
}
|
|
|
|
|
2020-05-09 21:22:51 -04:00
|
|
|
ClockingHint::Preference Drive::preferred_clocking() const {
|
2020-01-15 23:29:52 -05:00
|
|
|
return (!has_disk_ || (time_until_motor_transition == Cycles(0) && !disk_is_rotating_)) ? ClockingHint::Preference::None : ClockingHint::Preference::JustInTime;
|
2017-08-20 11:54:54 -04:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::get_is_track_zero() const {
|
2018-05-06 23:17:36 -04:00
|
|
|
return head_position_ == HeadPosition(0);
|
2016-09-25 21:24:16 -04:00
|
|
|
}
|
|
|
|
|
2018-05-06 23:17:36 -04:00
|
|
|
void Drive::step(HeadPosition offset) {
|
2020-07-17 22:08:58 -04:00
|
|
|
if(offset == HeadPosition(0)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-16 21:34:48 -05:00
|
|
|
if(ready_type_ == ReadyType::IBMRDY) {
|
|
|
|
is_ready_ = true;
|
|
|
|
}
|
|
|
|
|
2018-05-06 23:17:36 -04:00
|
|
|
HeadPosition old_head_position = head_position_;
|
|
|
|
head_position_ += offset;
|
2018-05-10 21:58:14 -04:00
|
|
|
if(head_position_ < HeadPosition(0)) {
|
|
|
|
head_position_ = HeadPosition(0);
|
|
|
|
if(observer_) observer_->announce_drive_event(drive_name_, Activity::Observer::DriveEvent::StepBelowZero);
|
2018-05-11 21:44:08 -04:00
|
|
|
} else {
|
|
|
|
if(observer_) observer_->announce_drive_event(drive_name_, Activity::Observer::DriveEvent::StepNormal);
|
2018-05-10 21:58:14 -04:00
|
|
|
}
|
2017-09-10 14:43:20 -04:00
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
// If the head moved, flush the old track.
|
2020-07-17 22:08:58 -04:00
|
|
|
if(disk_ && disk_->tracks_differ(Track::Address(head_, head_position_), Track::Address(head_, old_head_position))) {
|
2017-09-10 17:33:01 -04:00
|
|
|
track_ = nullptr;
|
|
|
|
}
|
2019-06-04 21:41:54 -04:00
|
|
|
|
|
|
|
// Allow a subclass to react, if desired.
|
|
|
|
did_step(head_position_);
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
|
2018-06-09 12:51:53 -04:00
|
|
|
std::shared_ptr<Track> Drive::step_to(HeadPosition offset) {
|
|
|
|
HeadPosition old_head_position = head_position_;
|
|
|
|
head_position_ = std::max(offset, HeadPosition(0));
|
|
|
|
|
2019-10-26 22:57:05 -04:00
|
|
|
if(disk_ && head_position_ != old_head_position) {
|
2018-06-09 12:51:53 -04:00
|
|
|
track_ = nullptr;
|
|
|
|
setup_track();
|
|
|
|
}
|
|
|
|
|
|
|
|
return track_;
|
|
|
|
}
|
|
|
|
|
2017-10-06 21:45:12 -04:00
|
|
|
void Drive::set_head(int head) {
|
2017-09-15 21:18:36 -04:00
|
|
|
head = std::min(head, available_heads_ - 1);
|
2017-09-10 17:33:01 -04:00
|
|
|
if(head != head_) {
|
|
|
|
head_ = head;
|
2017-09-10 14:43:20 -04:00
|
|
|
track_ = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
int Drive::get_head_count() const {
|
2019-06-04 21:41:54 -04:00
|
|
|
return available_heads_;
|
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::get_tachometer() const {
|
2019-06-06 21:36:19 -04:00
|
|
|
// I have made a guess here that the tachometer is a symmetric square wave;
|
|
|
|
// if that is correct then around 60 beats per rotation appears to be correct
|
|
|
|
// to proceed beyond the speed checks I've so far uncovered.
|
2019-12-22 00:22:17 -05:00
|
|
|
constexpr float ticks_per_rotation = 60.0f; // 56 was too low; 64 too high.
|
2019-06-06 21:36:19 -04:00
|
|
|
return int(get_rotation() * 2.0f * ticks_per_rotation) & 1;
|
2019-06-06 18:32:11 -04:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
float Drive::get_rotation() const {
|
2019-07-02 15:43:03 -04:00
|
|
|
return get_time_into_track();
|
2019-06-06 18:32:11 -04:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
float Drive::get_time_into_track() const {
|
2019-07-02 15:43:03 -04:00
|
|
|
// i.e. amount of time since the index hole was seen, as a proportion of a second,
|
|
|
|
// converted to a proportion of a rotation.
|
|
|
|
return float(cycles_since_index_hole_) / (float(get_input_clock_rate()) * rotational_multiplier_);
|
2016-09-25 21:24:16 -04:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::get_is_read_only() const {
|
2016-12-24 22:11:31 -05:00
|
|
|
if(disk_) return disk_->get_is_read_only();
|
2017-08-13 11:50:49 -04:00
|
|
|
return true;
|
2016-12-24 22:11:31 -05:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::get_is_ready() const {
|
2020-01-16 21:34:48 -05:00
|
|
|
return is_ready_;
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Drive::set_motor_on(bool motor_is_on) {
|
2020-01-15 23:29:52 -05:00
|
|
|
// Do nothing if the input hasn't changed.
|
|
|
|
if(motor_input_is_on_ == motor_is_on) return;
|
|
|
|
motor_input_is_on_ = motor_is_on;
|
|
|
|
|
|
|
|
// If this now means that the input and the actual state are in harmony,
|
|
|
|
// cancel any planned change and stop.
|
|
|
|
if(disk_is_rotating_ == motor_is_on) {
|
|
|
|
time_until_motor_transition = Cycles(0);
|
|
|
|
return;
|
|
|
|
}
|
2018-05-10 21:54:10 -04:00
|
|
|
|
2020-01-15 23:29:52 -05:00
|
|
|
// If this is a transition to on, start immediately.
|
|
|
|
// TODO: spin-up?
|
|
|
|
// TODO: momentum.
|
|
|
|
if(motor_is_on) {
|
|
|
|
set_disk_is_rotating(true);
|
2020-02-11 21:59:13 -05:00
|
|
|
time_until_motor_transition = Cycles(0);
|
2020-01-15 23:29:52 -05:00
|
|
|
return;
|
2017-09-11 22:27:50 -04:00
|
|
|
}
|
2020-01-15 23:29:52 -05:00
|
|
|
|
|
|
|
// This is a transition from on to off. Simulate momentum (ha!)
|
|
|
|
// by delaying the time until complete standstill.
|
|
|
|
if(time_until_motor_transition == Cycles(0))
|
|
|
|
time_until_motor_transition = get_input_clock_rate();
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::get_motor_on() const {
|
2020-01-15 23:16:25 -05:00
|
|
|
return motor_input_is_on_;
|
2017-09-10 19:23:23 -04:00
|
|
|
}
|
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::get_index_pulse() const {
|
|
|
|
return index_pulse_remaining_ > Cycles(0);
|
|
|
|
}
|
|
|
|
|
2017-09-10 19:23:23 -04:00
|
|
|
void Drive::set_event_delegate(Storage::Disk::Drive::EventDelegate *delegate) {
|
|
|
|
event_delegate_ = delegate;
|
|
|
|
}
|
|
|
|
|
2017-09-10 20:51:05 -04:00
|
|
|
void Drive::advance(const Cycles cycles) {
|
2019-10-29 22:36:29 -04:00
|
|
|
cycles_since_index_hole_ += cycles.as_integral();
|
2017-09-10 20:51:05 -04:00
|
|
|
if(event_delegate_) event_delegate_->advance(cycles);
|
|
|
|
}
|
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
void Drive::run_for(const Cycles cycles) {
|
2019-12-24 21:05:17 -05:00
|
|
|
// Assumed: the index pulse pulses even if the drive has stopped spinning.
|
|
|
|
index_pulse_remaining_ = std::max(index_pulse_remaining_ - cycles, Cycles(0));
|
|
|
|
|
2020-01-15 23:29:52 -05:00
|
|
|
if(time_until_motor_transition > Cycles(0)) {
|
|
|
|
if(time_until_motor_transition > cycles) {
|
|
|
|
time_until_motor_transition -= cycles;
|
|
|
|
} else {
|
|
|
|
time_until_motor_transition = Cycles(0);
|
|
|
|
set_disk_is_rotating(!disk_is_rotating_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(disk_is_rotating_) {
|
2019-10-26 22:57:05 -04:00
|
|
|
if(has_disk_) {
|
|
|
|
Time zero(0);
|
|
|
|
|
2019-10-29 22:36:29 -04:00
|
|
|
auto number_of_cycles = cycles.as_integral();
|
2019-10-26 22:57:05 -04:00
|
|
|
while(number_of_cycles) {
|
2019-10-29 22:36:29 -04:00
|
|
|
auto cycles_until_next_event = get_cycles_until_next_event();
|
|
|
|
auto cycles_to_run_for = std::min(cycles_until_next_event, number_of_cycles);
|
2019-10-26 22:57:05 -04:00
|
|
|
if(!is_reading_ && cycles_until_bits_written_ > zero) {
|
2019-10-29 22:36:29 -04:00
|
|
|
auto write_cycles_target = cycles_until_bits_written_.get<Cycles::IntType>();
|
|
|
|
if(cycles_until_bits_written_.length % cycles_until_bits_written_.clock_rate) ++write_cycles_target;
|
2019-10-26 22:57:05 -04:00
|
|
|
cycles_to_run_for = std::min(cycles_to_run_for, write_cycles_target);
|
|
|
|
}
|
2017-09-10 17:33:01 -04:00
|
|
|
|
2019-10-26 22:57:05 -04:00
|
|
|
number_of_cycles -= cycles_to_run_for;
|
|
|
|
if(!is_reading_) {
|
|
|
|
if(cycles_until_bits_written_ > zero) {
|
2019-10-29 22:36:29 -04:00
|
|
|
Storage::Time cycles_to_run_for_time(static_cast<int>(cycles_to_run_for));
|
2019-10-26 22:57:05 -04:00
|
|
|
if(cycles_until_bits_written_ <= cycles_to_run_for_time) {
|
|
|
|
if(event_delegate_) event_delegate_->process_write_completed();
|
|
|
|
if(cycles_until_bits_written_ <= cycles_to_run_for_time)
|
|
|
|
cycles_until_bits_written_.set_zero();
|
|
|
|
else
|
|
|
|
cycles_until_bits_written_ -= cycles_to_run_for_time;
|
|
|
|
} else {
|
2017-09-10 17:33:01 -04:00
|
|
|
cycles_until_bits_written_ -= cycles_to_run_for_time;
|
2019-10-26 22:57:05 -04:00
|
|
|
}
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
}
|
2019-10-26 22:57:05 -04:00
|
|
|
TimedEventLoop::run_for(Cycles(cycles_to_run_for));
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
2019-10-26 22:57:05 -04:00
|
|
|
} else {
|
|
|
|
TimedEventLoop::run_for(cycles);
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-12 15:59:11 -05:00
|
|
|
// MARK: - Track timed event loop
|
2017-09-10 17:33:01 -04:00
|
|
|
|
2019-07-02 15:43:03 -04:00
|
|
|
void Drive::get_next_event(float duration_already_passed) {
|
2020-07-15 19:34:05 -04:00
|
|
|
/*
|
|
|
|
Quick word on random-bit generation logic below; it seeks to obey the following logic:
|
|
|
|
if there is a gap of 15µs between recorded bits, start generating flux transitions
|
|
|
|
at random intervals thereafter, unless and until one is within 5µs of the next real transition.
|
|
|
|
|
|
|
|
This behaviour is based on John Morris' observations of an MC3470, as described in his WOZ
|
|
|
|
file format documentation — https://applesaucefdc.com/woz/reference2/
|
|
|
|
*/
|
|
|
|
|
2019-10-26 22:57:05 -04:00
|
|
|
if(!disk_) {
|
|
|
|
current_event_.type = Track::Event::IndexHole;
|
|
|
|
current_event_.length = 1.0f;
|
|
|
|
set_next_event_time_interval((current_event_.length - duration_already_passed) * rotational_multiplier_);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-09-10 22:44:14 -04:00
|
|
|
// Grab a new track if not already in possession of one. This will recursively call get_next_event,
|
|
|
|
// supplying a proper duration_already_passed.
|
|
|
|
if(!track_) {
|
2019-07-02 15:43:03 -04:00
|
|
|
random_interval_ = 0.0f;
|
2017-09-10 22:44:14 -04:00
|
|
|
setup_track();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-14 19:17:34 -04:00
|
|
|
// If gain has now been turned up so as to generate noise, generate some noise.
|
2019-07-02 15:43:03 -04:00
|
|
|
if(random_interval_ > 0.0f) {
|
|
|
|
current_event_.type = Track::Event::FluxTransition;
|
2020-07-15 19:34:05 -04:00
|
|
|
current_event_.length = float(2 + (random_source_&1)) / 1'000'000.0f;
|
2018-05-14 19:17:34 -04:00
|
|
|
random_source_ = (random_source_ >> 1) | (random_source_ << 63);
|
|
|
|
|
2020-07-15 19:34:05 -04:00
|
|
|
// If this random transition is closer than 5µs to the next real bit,
|
|
|
|
// discard it.
|
2020-07-15 22:44:54 -04:00
|
|
|
if(random_interval_ - 5.0f / 1'000'000.f < current_event_.length) {
|
2019-07-02 15:43:03 -04:00
|
|
|
random_interval_ = 0.0f;
|
2018-05-14 19:17:34 -04:00
|
|
|
} else {
|
|
|
|
random_interval_ -= current_event_.length;
|
2020-07-15 19:34:05 -04:00
|
|
|
set_next_event_time_interval(current_event_.length);
|
|
|
|
return;
|
2018-05-14 19:17:34 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
if(track_) {
|
2019-07-02 15:43:03 -04:00
|
|
|
const auto track_event = track_->get_next_event();
|
|
|
|
current_event_.type = track_event.type;
|
|
|
|
current_event_.length = track_event.length.get<float>();
|
2017-09-10 17:33:01 -04:00
|
|
|
} else {
|
2019-07-02 15:43:03 -04:00
|
|
|
current_event_.length = 1.0f;
|
2017-09-10 17:33:01 -04:00
|
|
|
current_event_.type = Track::Event::IndexHole;
|
|
|
|
}
|
|
|
|
|
|
|
|
// divide interval, which is in terms of a single rotation of the disk, by rotation speed to
|
|
|
|
// convert it into revolutions per second; this is achieved by multiplying by rotational_multiplier_
|
2019-07-02 15:43:03 -04:00
|
|
|
float interval = std::max((current_event_.length - duration_already_passed) * rotational_multiplier_, 0.0f);
|
2018-05-14 19:17:34 -04:00
|
|
|
|
2020-07-15 19:34:05 -04:00
|
|
|
// An interval greater than 15µs => adjust gain up the point where noise starts happening.
|
|
|
|
// Seed that up and leave a 15µs gap until it starts.
|
|
|
|
constexpr float safe_gain_period = 15.0f / 1'000'000.0f;
|
2018-05-14 19:17:34 -04:00
|
|
|
if(interval >= safe_gain_period) {
|
|
|
|
random_interval_ = interval - safe_gain_period;
|
|
|
|
interval = safe_gain_period;
|
|
|
|
}
|
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
set_next_event_time_interval(interval);
|
2017-08-08 21:15:56 -04:00
|
|
|
}
|
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
void Drive::process_next_event() {
|
2017-09-10 22:44:14 -04:00
|
|
|
if(current_event_.type == Track::Event::IndexHole) {
|
2020-01-16 21:34:48 -05:00
|
|
|
++ready_index_count_;
|
|
|
|
if(ready_index_count_ == 2 && (ready_type_ == ReadyType::ShugartRDY || ready_type_ == ReadyType::ShugartModifiedRDY)) {
|
|
|
|
is_ready_ = true;
|
|
|
|
}
|
2017-09-10 22:44:14 -04:00
|
|
|
cycles_since_index_hole_ = 0;
|
2020-07-19 00:06:27 -04:00
|
|
|
|
|
|
|
// Begin a 2ms period of holding the index line pulse active.
|
|
|
|
index_pulse_remaining_ = Cycles((get_input_clock_rate() * 2) / 1000);
|
2017-09-10 22:44:14 -04:00
|
|
|
}
|
2017-09-14 22:32:13 -04:00
|
|
|
if(
|
|
|
|
event_delegate_ &&
|
|
|
|
(current_event_.type == Track::Event::IndexHole || is_reading_)
|
|
|
|
){
|
|
|
|
event_delegate_->process_event(current_event_);
|
|
|
|
}
|
2019-07-02 15:43:03 -04:00
|
|
|
get_next_event(0.0f);
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
|
2017-11-12 15:59:11 -05:00
|
|
|
// MARK: - Track management
|
2017-09-10 17:33:01 -04:00
|
|
|
|
2017-03-26 14:34:47 -04:00
|
|
|
std::shared_ptr<Track> Drive::get_track() {
|
2017-10-06 21:45:12 -04:00
|
|
|
if(disk_) return disk_->get_track_at_position(Track::Address(head_, head_position_));
|
2016-09-25 21:24:16 -04:00
|
|
|
return nullptr;
|
|
|
|
}
|
2016-12-24 22:11:31 -05:00
|
|
|
|
2017-03-26 14:34:47 -04:00
|
|
|
void Drive::set_track(const std::shared_ptr<Track> &track) {
|
2017-10-06 21:45:12 -04:00
|
|
|
if(disk_) disk_->set_track_at_position(Track::Address(head_, head_position_), track);
|
2016-12-24 22:11:31 -05:00
|
|
|
}
|
2017-09-10 14:43:20 -04:00
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
void Drive::setup_track() {
|
|
|
|
track_ = get_track();
|
|
|
|
if(!track_) {
|
2019-12-23 21:31:46 -05:00
|
|
|
track_ = std::make_shared<UnformattedTrack>();
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
|
2019-07-02 15:43:03 -04:00
|
|
|
float offset = 0.0f;
|
2020-07-17 22:08:58 -04:00
|
|
|
const float track_time_now = get_time_into_track();
|
|
|
|
const float time_found = track_->seek_to(track_time_now);
|
2017-09-10 17:33:01 -04:00
|
|
|
|
2019-07-02 15:43:03 -04:00
|
|
|
// `time_found` can be greater than `track_time_now` if limited precision caused rounding.
|
2018-05-02 21:26:39 -04:00
|
|
|
if(time_found <= track_time_now) {
|
|
|
|
offset = track_time_now - time_found;
|
|
|
|
}
|
|
|
|
|
2019-07-26 23:23:01 -04:00
|
|
|
// Reseed cycles_since_index_hole_; 99.99% of the time it'll still be correct as is,
|
|
|
|
// but if the track has rounded one way or the other it may now be very slightly adrift.
|
|
|
|
cycles_since_index_hole_ = (int((time_found + offset) * cycles_per_revolution_)) % cycles_per_revolution_;
|
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
get_next_event(offset);
|
2017-09-10 14:43:20 -04:00
|
|
|
}
|
|
|
|
|
2017-09-10 17:33:01 -04:00
|
|
|
void Drive::invalidate_track() {
|
2019-07-02 15:43:03 -04:00
|
|
|
random_interval_ = 0.0f;
|
2017-09-10 17:33:01 -04:00
|
|
|
track_ = nullptr;
|
2017-09-10 19:23:23 -04:00
|
|
|
if(patched_track_) {
|
|
|
|
set_track(patched_track_);
|
|
|
|
patched_track_ = nullptr;
|
|
|
|
}
|
2017-09-10 14:43:20 -04:00
|
|
|
}
|
|
|
|
|
2017-11-12 15:59:11 -05:00
|
|
|
// MARK: - Writing
|
2017-09-10 17:33:01 -04:00
|
|
|
|
|
|
|
void Drive::begin_writing(Time bit_length, bool clamp_to_index_hole) {
|
2019-07-12 18:53:41 -04:00
|
|
|
// Do nothing if already writing.
|
2019-10-26 22:57:05 -04:00
|
|
|
// TODO: cope properly if there's no disk to write to.
|
|
|
|
if(!is_reading_ || !disk_) return;
|
2019-07-12 18:53:41 -04:00
|
|
|
|
|
|
|
// Get a copy of the track if that hasn't happened yet.
|
|
|
|
if(!track_) {
|
|
|
|
setup_track();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store the relevant parameters, and kick off writing.
|
2017-09-10 17:33:01 -04:00
|
|
|
is_reading_ = false;
|
|
|
|
clamp_writing_to_index_hole_ = clamp_to_index_hole;
|
|
|
|
|
2019-10-29 22:36:29 -04:00
|
|
|
cycles_per_bit_ = Storage::Time(int(get_input_clock_rate())) * bit_length;
|
2017-09-10 19:23:23 -04:00
|
|
|
cycles_per_bit_.simplify();
|
|
|
|
|
2019-07-02 15:43:03 -04:00
|
|
|
write_segment_.length_of_a_bit = bit_length / Time(rotational_multiplier_);
|
2017-09-10 17:33:01 -04:00
|
|
|
write_segment_.data.clear();
|
|
|
|
|
2019-07-02 15:43:03 -04:00
|
|
|
write_start_time_ = Time(get_time_into_track());
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Drive::write_bit(bool value) {
|
2018-07-01 12:05:41 -04:00
|
|
|
write_segment_.data.push_back(value);
|
2017-09-10 17:33:01 -04:00
|
|
|
cycles_until_bits_written_ += cycles_per_bit_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Drive::end_writing() {
|
2018-07-02 21:51:53 -04:00
|
|
|
// If the user modifies a track, it's scaled up to a "high" resolution and modifications
|
|
|
|
// are plotted on top of that.
|
2019-07-26 15:26:51 -04:00
|
|
|
//
|
|
|
|
// "High" is defined as: two samples per clock relative to an idiomatic
|
|
|
|
// 8Mhz disk controller and 300RPM disk speed.
|
|
|
|
const size_t high_resolution_track_rate = 3200000;
|
2018-07-01 22:49:57 -04:00
|
|
|
|
2017-09-16 17:07:36 -04:00
|
|
|
if(!is_reading_) {
|
|
|
|
is_reading_ = true;
|
2017-09-10 17:33:01 -04:00
|
|
|
|
|
|
|
if(!patched_track_) {
|
2017-09-16 17:07:36 -04:00
|
|
|
// Avoid creating a new patched track if this one is already patched
|
2018-07-01 22:49:57 -04:00
|
|
|
patched_track_ = std::dynamic_pointer_cast<PCMTrack>(track_);
|
|
|
|
if(!patched_track_ || !patched_track_->is_resampled_clone()) {
|
2019-07-15 22:40:45 -04:00
|
|
|
Track *const tr = track_.get();
|
2018-07-01 22:49:57 -04:00
|
|
|
patched_track_.reset(PCMTrack::resampled_clone(tr, high_resolution_track_rate));
|
2017-09-16 17:07:36 -04:00
|
|
|
}
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
2017-09-16 17:07:36 -04:00
|
|
|
patched_track_->add_segment(write_start_time_, write_segment_, clamp_writing_to_index_hole_);
|
2019-07-26 17:20:32 -04:00
|
|
|
cycles_since_index_hole_ %= cycles_per_revolution_;
|
2017-09-16 17:07:36 -04:00
|
|
|
invalidate_track();
|
2017-09-10 17:33:01 -04:00
|
|
|
}
|
2017-09-10 14:43:20 -04:00
|
|
|
}
|
2018-05-10 21:54:10 -04:00
|
|
|
|
2019-12-24 20:53:37 -05:00
|
|
|
bool Drive::is_writing() const {
|
2019-07-12 18:53:41 -04:00
|
|
|
return !is_reading_;
|
|
|
|
}
|
|
|
|
|
2020-01-15 23:29:52 -05:00
|
|
|
void Drive::set_disk_is_rotating(bool is_rotating) {
|
|
|
|
disk_is_rotating_ = is_rotating;
|
|
|
|
|
|
|
|
if(observer_) {
|
2020-02-11 21:59:13 -05:00
|
|
|
observer_->set_drive_motor_status(drive_name_, disk_is_rotating_);
|
2020-01-15 23:29:52 -05:00
|
|
|
if(announce_motor_led_) {
|
2020-02-11 21:59:13 -05:00
|
|
|
observer_->set_led_status(drive_name_, disk_is_rotating_);
|
2020-01-15 23:29:52 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!is_rotating) {
|
2020-01-16 21:34:48 -05:00
|
|
|
if(ready_type_ == ReadyType::ShugartRDY) {
|
|
|
|
is_ready_ = false;
|
|
|
|
}
|
2020-01-15 23:29:52 -05:00
|
|
|
ready_index_count_ = 0;
|
|
|
|
if(disk_) disk_->flush_tracks();
|
|
|
|
}
|
|
|
|
update_clocking_observer();
|
|
|
|
}
|
|
|
|
|
2018-05-10 21:54:10 -04:00
|
|
|
void Drive::set_activity_observer(Activity::Observer *observer, const std::string &name, bool add_motor_led) {
|
|
|
|
observer_ = observer;
|
|
|
|
announce_motor_led_ = add_motor_led;
|
|
|
|
if(observer) {
|
|
|
|
drive_name_ = name;
|
|
|
|
|
|
|
|
observer->register_drive(drive_name_);
|
2020-01-15 23:29:52 -05:00
|
|
|
observer->set_drive_motor_status(drive_name_, disk_is_rotating_);
|
2018-05-10 21:54:10 -04:00
|
|
|
|
|
|
|
if(add_motor_led) {
|
|
|
|
observer->register_led(drive_name_);
|
2020-01-15 23:29:52 -05:00
|
|
|
observer->set_led_status(drive_name_, disk_is_rotating_);
|
2018-05-10 21:54:10 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|