mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-11 08:30:55 +00:00
150 lines
5.2 KiB
C++
150 lines
5.2 KiB
C++
//
|
|
// Disk.hpp
|
|
// Clock Signal
|
|
//
|
|
// Created by Thomas Harte on 10/07/2016.
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#ifndef Disk_hpp
|
|
#define Disk_hpp
|
|
|
|
#include <map>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <set>
|
|
|
|
#include "../Storage.hpp"
|
|
#include "../../Concurrency/AsyncTaskQueue.hpp"
|
|
|
|
namespace Storage {
|
|
namespace Disk {
|
|
|
|
/*!
|
|
Models a single track on a disk as a series of events, each event being of arbitrary length
|
|
and resulting in either a flux transition or the sensing of an index hole.
|
|
|
|
Subclasses should implement @c get_next_event.
|
|
*/
|
|
class Track {
|
|
public:
|
|
/*!
|
|
Describes a detectable track event — either a flux transition or the passing of the index hole,
|
|
along with the length of time between the previous event and its occurance.
|
|
|
|
The sum of all lengths of time across an entire track should be 1 — if an event is said to be
|
|
1/3 away then that means 1/3 of a rotation.
|
|
*/
|
|
struct Event {
|
|
enum {
|
|
IndexHole, FluxTransition
|
|
} type;
|
|
Time length;
|
|
};
|
|
|
|
/*!
|
|
@returns the next event that will be detected during rotation of this disk.
|
|
*/
|
|
virtual Event get_next_event() = 0;
|
|
|
|
/*!
|
|
Jumps to the event latest offset that is less than or equal to the input time.
|
|
|
|
@returns the time jumped to.
|
|
*/
|
|
virtual Time seek_to(const Time &time_since_index_hole) = 0;
|
|
|
|
/*!
|
|
The virtual copy constructor pattern; returns a copy of the Track.
|
|
*/
|
|
virtual Track *clone() = 0;
|
|
};
|
|
|
|
/*!
|
|
Models a disk as a collection of tracks, providing a range of possible track positions and allowing
|
|
a point sampling of the track beneath any of those positions (if any).
|
|
|
|
The intention is not that tracks necessarily be evenly spaced; a head_position_count of 3 wih track
|
|
A appearing in positions 0 and 1, and track B appearing in position 2 is an appropriate use of this API
|
|
if it matches the media.
|
|
|
|
The track returned is point sampled only; if a particular disk drive has a sufficiently large head to
|
|
pick up multiple tracks at once then the drive responsible for asking for multiple tracks and for
|
|
merging the results.
|
|
*/
|
|
class Disk {
|
|
public:
|
|
virtual ~Disk() {}
|
|
|
|
/*!
|
|
@returns the number of discrete positions that this disk uses to model its complete surface area.
|
|
|
|
This is not necessarily a track count. There is no implicit guarantee that every position will
|
|
return a distinct track, or — e.g. if the media is holeless — will return any track at all.
|
|
*/
|
|
virtual unsigned int get_head_position_count() = 0;
|
|
|
|
/*!
|
|
@returns the number of heads (and, therefore, impliedly surfaces) available on this disk.
|
|
*/
|
|
virtual unsigned int get_head_count() { return 1; }
|
|
|
|
/*!
|
|
@returns the @c Track at @c position underneath @c head if there are any detectable events there;
|
|
returns @c nullptr otherwise.
|
|
*/
|
|
std::shared_ptr<Track> get_track_at_position(unsigned int head, unsigned int position);
|
|
|
|
/*!
|
|
Replaces the Track at position @c position underneath @c head with @c track. Ignored if this disk is read-only.
|
|
Subclasses that are not read-only should use the protected methods @c get_is_modified and, optionally,
|
|
@c get_modified_track_at_position to query for changes when closing.
|
|
*/
|
|
void set_track_at_position(unsigned int head, unsigned int position, const std::shared_ptr<Track> &track);
|
|
|
|
/*!
|
|
@returns whether the disk image is read only. Defaults to @c true if not overridden.
|
|
*/
|
|
virtual bool get_is_read_only() { return true; }
|
|
|
|
protected:
|
|
/*!
|
|
Subclasses should implement this to return the @c Track at @c position underneath @c head. Returned tracks
|
|
are cached internally so subclasses shouldn't attempt to build their own caches or worry about preparing
|
|
for track accesses at file load time. Appropriate behaviour is to create them lazily, on demand.
|
|
*/
|
|
virtual std::shared_ptr<Track> get_uncached_track_at_position(unsigned int head, unsigned int position) = 0;
|
|
|
|
/*!
|
|
Subclasses that support writing should implement @c store_updated_track_at_position to determine which bytes
|
|
have to be written from @c track, then obtain @c file_access_mutex and write new data to their file to represent
|
|
the track underneath @c head at @c position.
|
|
|
|
The base class will ensure that calls are made to @c get_uncached_track_at_position only while it holds @c file_access_mutex;
|
|
that mutex therefore provides serialisation of file access.
|
|
|
|
This method will be called asynchronously. Subclasses are responsible for any synchronisation other than that
|
|
provided automatically via @c file_access_mutex.
|
|
*/
|
|
virtual void store_updated_track_at_position(unsigned int head, unsigned int position, const std::shared_ptr<Track> &track, std::mutex &file_access_mutex);
|
|
|
|
/*!
|
|
Subclasses that support writing should call @c flush_updates during their destructor if there is anything they
|
|
do in @c store_updated_track_at_position that would not be valid after their destructor has completed but prior
|
|
to Disk's constructor running.
|
|
*/
|
|
void flush_updates();
|
|
|
|
private:
|
|
int get_id_for_track_at_position(unsigned int head, unsigned int position);
|
|
std::map<int, std::shared_ptr<Track>> cached_tracks_;
|
|
|
|
std::mutex file_access_mutex_;
|
|
std::unique_ptr<Concurrency::AsyncTaskQueue> update_queue_;
|
|
};
|
|
|
|
}
|
|
}
|
|
|
|
#endif /* Disk_hpp */
|