2016-09-26 01:24:16 +00:00
|
|
|
//
|
|
|
|
// Drive.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 25/09/2016.
|
2018-05-13 19:19:52 +00:00
|
|
|
// Copyright 2016 Thomas Harte. All rights reserved.
|
2016-09-26 01:24:16 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef Drive_hpp
|
|
|
|
#define Drive_hpp
|
|
|
|
|
|
|
|
#include "Disk.hpp"
|
2017-09-23 02:39:23 +00:00
|
|
|
#include "Track/PCMSegment.hpp"
|
2018-07-02 02:49:57 +00:00
|
|
|
#include "Track/PCMTrack.hpp"
|
2017-09-10 20:23:31 +00:00
|
|
|
|
2017-09-10 18:43:20 +00:00
|
|
|
#include "../TimedEventLoop.hpp"
|
2018-05-11 01:54:10 +00:00
|
|
|
#include "../../Activity/Observer.hpp"
|
2018-05-28 03:17:06 +00:00
|
|
|
#include "../../ClockReceiver/ClockingHintSource.hpp"
|
2017-09-10 20:23:31 +00:00
|
|
|
|
|
|
|
#include <memory>
|
2016-09-26 01:24:16 +00:00
|
|
|
|
|
|
|
namespace Storage {
|
|
|
|
namespace Disk {
|
|
|
|
|
2018-05-28 03:17:06 +00:00
|
|
|
class Drive: public ClockingHint::Source, public TimedEventLoop {
|
2016-09-26 01:24:16 +00:00
|
|
|
public:
|
2017-10-07 01:45:12 +00:00
|
|
|
Drive(unsigned int input_clock_rate, int revolutions_per_minute, int number_of_heads);
|
2017-10-07 23:14:18 +00:00
|
|
|
~Drive();
|
2016-09-26 01:24:16 +00:00
|
|
|
|
|
|
|
/*!
|
2016-12-26 19:24:33 +00:00
|
|
|
Replaces whatever is in the drive with @c disk.
|
2016-09-26 01:24:16 +00:00
|
|
|
*/
|
2016-12-26 19:24:33 +00:00
|
|
|
void set_disk(const std::shared_ptr<Disk> &disk);
|
|
|
|
|
2016-09-26 01:24:16 +00:00
|
|
|
/*!
|
|
|
|
@returns @c true if a disk is currently inserted; @c false otherwise.
|
|
|
|
*/
|
|
|
|
bool has_disk();
|
|
|
|
|
|
|
|
/*!
|
|
|
|
@returns @c true if the drive head is currently at track zero; @c false otherwise.
|
|
|
|
*/
|
|
|
|
bool get_is_track_zero();
|
|
|
|
|
|
|
|
/*!
|
2016-12-25 03:37:20 +00:00
|
|
|
Steps the disk head the specified number of tracks. Positive numbers step inwards (i.e. away from track 0),
|
|
|
|
negative numbers step outwards (i.e. towards track 0).
|
2016-09-26 01:24:16 +00:00
|
|
|
*/
|
2018-05-07 03:17:36 +00:00
|
|
|
void step(HeadPosition offset);
|
2016-09-26 01:24:16 +00:00
|
|
|
|
|
|
|
/*!
|
2016-12-25 03:37:20 +00:00
|
|
|
Sets the current read head.
|
2016-09-26 01:24:16 +00:00
|
|
|
*/
|
2017-10-07 01:45:12 +00:00
|
|
|
void set_head(int head);
|
2016-09-26 01:24:16 +00:00
|
|
|
|
2016-12-25 03:37:20 +00:00
|
|
|
/*!
|
2017-08-13 15:50:49 +00:00
|
|
|
@returns @c true if the inserted disk is read-only or no disk is inserted; @c false otherwise.
|
2016-12-25 03:37:20 +00:00
|
|
|
*/
|
2016-12-25 03:11:31 +00:00
|
|
|
bool get_is_read_only();
|
|
|
|
|
2017-08-09 01:15:56 +00:00
|
|
|
/*!
|
|
|
|
@returns @c true if the drive is ready; @c false otherwise.
|
|
|
|
*/
|
|
|
|
bool get_is_ready();
|
|
|
|
|
2017-09-10 18:43:20 +00:00
|
|
|
/*!
|
|
|
|
Sets whether the disk motor is on.
|
|
|
|
*/
|
|
|
|
void set_motor_on(bool);
|
|
|
|
|
2017-09-10 20:23:31 +00:00
|
|
|
/*!
|
|
|
|
@returns @c true if the motor is on; @c false otherwise.
|
|
|
|
*/
|
|
|
|
bool get_motor_on();
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Begins write mode, initiating a PCM sampled region of data. Bits should be written via
|
|
|
|
@c write_bit. They will be written with the length set via @c set_expected_bit_length.
|
|
|
|
It is acceptable to supply a backlog of bits. Flux transition events will not be reported
|
|
|
|
while writing.
|
|
|
|
|
|
|
|
@param clamp_to_index_hole If @c true then writing will automatically be truncated by
|
|
|
|
the index hole. Writing will continue over the index hole otherwise.
|
|
|
|
*/
|
2017-09-10 21:33:01 +00:00
|
|
|
void begin_writing(Time bit_length, bool clamp_to_index_hole);
|
2017-09-10 20:23:31 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
Writes the bit @c value as the next in the PCM stream initiated by @c begin_writing.
|
|
|
|
*/
|
|
|
|
void write_bit(bool value);
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Ends write mode, switching back to read mode. The drive will stop overwriting events.
|
|
|
|
*/
|
|
|
|
void end_writing();
|
|
|
|
|
2017-09-10 18:43:20 +00:00
|
|
|
/*!
|
|
|
|
Advances the drive by @c number_of_cycles cycles.
|
|
|
|
*/
|
|
|
|
void run_for(const Cycles cycles);
|
|
|
|
|
|
|
|
/*!
|
2017-09-10 20:23:31 +00:00
|
|
|
Provides a mechanism to receive track events as they occur, including the synthetic
|
|
|
|
event of "you told me to output the following data, and I've done that now".
|
2017-09-10 18:43:20 +00:00
|
|
|
*/
|
|
|
|
struct EventDelegate {
|
2017-09-10 20:23:31 +00:00
|
|
|
/// Informs the delegate that @c event has been reached.
|
2017-09-10 18:43:20 +00:00
|
|
|
virtual void process_event(const Track::Event &event) = 0;
|
2017-09-10 20:23:31 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
If the drive is in write mode, announces that all queued bits have now been written.
|
|
|
|
If the controller provides further bits now then there will be no gap in written data.
|
|
|
|
*/
|
2018-04-25 02:44:45 +00:00
|
|
|
virtual void process_write_completed() {}
|
2017-09-10 21:33:01 +00:00
|
|
|
|
|
|
|
/// Informs the delegate of the passing of @c cycles.
|
2018-04-25 02:44:45 +00:00
|
|
|
virtual void advance(const Cycles cycles) {}
|
2017-09-10 18:43:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// Sets the current event delegate.
|
|
|
|
void set_event_delegate(EventDelegate *);
|
|
|
|
|
2017-08-20 15:54:54 +00:00
|
|
|
// As per Sleeper.
|
2018-05-28 03:17:06 +00:00
|
|
|
ClockingHint::Preference preferred_clocking() override;
|
2017-08-20 15:54:54 +00:00
|
|
|
|
2018-05-11 01:54:10 +00:00
|
|
|
/// Adds an activity observer; it'll be notified of disk activity.
|
|
|
|
/// The caller can specify whether to add an LED based on disk motor.
|
|
|
|
void set_activity_observer(Activity::Observer *observer, const std::string &name, bool add_motor_led);
|
|
|
|
|
2018-06-09 16:51:53 +00:00
|
|
|
/*!
|
|
|
|
Attempts to step to the specified offset and returns the track there if one exists; an uninitialised
|
|
|
|
track otherwise.
|
|
|
|
|
|
|
|
This is unambiguously **NOT A REALISTIC DRIVE FUNCTION**; real drives cannot step to a given offset.
|
|
|
|
So it is **NOT FOR HARDWARE EMULATION USAGE**.
|
|
|
|
|
|
|
|
It's for the benefit of user-optional fast-loading mechanisms **ONLY**.
|
|
|
|
*/
|
|
|
|
std::shared_ptr<Track> step_to(HeadPosition offset);
|
|
|
|
|
2016-09-26 01:24:16 +00:00
|
|
|
private:
|
2017-09-10 21:33:01 +00:00
|
|
|
// Drives contain an entire disk; from that a certain track
|
2017-09-10 20:23:31 +00:00
|
|
|
// will be currently under the head.
|
2016-12-03 16:59:28 +00:00
|
|
|
std::shared_ptr<Disk> disk_;
|
2017-09-10 20:23:31 +00:00
|
|
|
std::shared_ptr<Track> track_;
|
|
|
|
bool has_disk_ = false;
|
|
|
|
|
|
|
|
// Contains the multiplier that converts between track-relative lengths
|
2018-05-13 19:34:31 +00:00
|
|
|
// to real-time lengths. So it's the reciprocal of rotation speed.
|
2017-09-10 18:43:20 +00:00
|
|
|
Time rotational_multiplier_;
|
|
|
|
|
2017-09-10 20:23:31 +00:00
|
|
|
// A count of time since the index hole was last seen. Which is used to
|
|
|
|
// determine how far the drive is into a full rotation when switching to
|
|
|
|
// a new track.
|
|
|
|
int cycles_since_index_hole_ = 0;
|
2017-09-10 18:43:20 +00:00
|
|
|
|
2017-09-10 20:23:31 +00:00
|
|
|
// A record of head position and active head.
|
2018-05-07 03:17:36 +00:00
|
|
|
HeadPosition head_position_;
|
2017-10-07 01:45:12 +00:00
|
|
|
int head_ = 0;
|
|
|
|
int available_heads_ = 0;
|
2017-09-10 18:43:20 +00:00
|
|
|
|
2017-09-10 20:23:31 +00:00
|
|
|
// Motor control state.
|
|
|
|
bool motor_is_on_ = false;
|
|
|
|
|
|
|
|
// If the drive is not currently reading then it is writing. While writing
|
|
|
|
// it can optionally be told to clamp to the index hole.
|
2017-09-11 00:51:05 +00:00
|
|
|
bool is_reading_ = true;
|
2017-09-10 21:33:01 +00:00
|
|
|
bool clamp_writing_to_index_hole_ = false;
|
2017-09-10 20:23:31 +00:00
|
|
|
|
|
|
|
// If writing is occurring then the drive will be accumulating a write segment,
|
2018-07-02 02:49:57 +00:00
|
|
|
// for addition to a (high-resolution) PCM track.
|
|
|
|
std::shared_ptr<PCMTrack> patched_track_;
|
2017-09-10 20:23:31 +00:00
|
|
|
PCMSegment write_segment_;
|
|
|
|
Time write_start_time_;
|
|
|
|
|
2017-09-12 02:27:50 +00:00
|
|
|
// Indicates progress towards drive ready state.
|
|
|
|
int ready_index_count_ = 0;
|
2017-09-10 21:33:01 +00:00
|
|
|
|
2017-09-10 20:23:31 +00:00
|
|
|
// Maintains appropriate counting to know when to indicate that writing
|
|
|
|
// is complete.
|
|
|
|
Time cycles_until_bits_written_;
|
|
|
|
Time cycles_per_bit_;
|
|
|
|
|
|
|
|
// TimedEventLoop call-ins and state.
|
2018-05-28 03:17:06 +00:00
|
|
|
void process_next_event() override;
|
2017-09-10 18:43:20 +00:00
|
|
|
void get_next_event(const Time &duration_already_passed);
|
2018-05-28 03:17:06 +00:00
|
|
|
void advance(const Cycles cycles) override;
|
2017-09-10 18:43:20 +00:00
|
|
|
Track::Event current_event_;
|
|
|
|
|
2017-09-10 20:23:31 +00:00
|
|
|
// Helper for track changes.
|
2017-09-10 18:43:20 +00:00
|
|
|
Time get_time_into_track();
|
2017-09-10 20:23:31 +00:00
|
|
|
|
|
|
|
// The target (if any) for track events.
|
2017-09-10 18:43:20 +00:00
|
|
|
EventDelegate *event_delegate_ = nullptr;
|
2017-09-10 21:33:01 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
@returns the track underneath the current head at the location now stepped to.
|
|
|
|
*/
|
|
|
|
std::shared_ptr<Track> get_track();
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Attempts to set @c track as the track underneath the current head at the location now stepped to.
|
|
|
|
*/
|
|
|
|
void set_track(const std::shared_ptr<Track> &track);
|
|
|
|
|
|
|
|
void setup_track();
|
|
|
|
void invalidate_track();
|
2018-05-11 01:54:10 +00:00
|
|
|
|
|
|
|
// Activity observer description.
|
|
|
|
Activity::Observer *observer_ = nullptr;
|
|
|
|
std::string drive_name_;
|
|
|
|
bool announce_motor_led_ = false;
|
2018-05-14 23:17:34 +00:00
|
|
|
|
|
|
|
// A rotating random data source.
|
|
|
|
uint64_t random_source_;
|
|
|
|
Time random_interval_;
|
2016-09-26 01:24:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* Drive_hpp */
|