2016-09-17 18:01:00 -04:00
|
|
|
//
|
|
|
|
// 1770.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 17/09/2016.
|
|
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef _770_hpp
|
|
|
|
#define _770_hpp
|
|
|
|
|
2016-09-25 20:05:56 -04:00
|
|
|
#include "../../Storage/Disk/DiskController.hpp"
|
2016-12-26 16:46:26 -05:00
|
|
|
#include "../../NumberTheory/CRC.hpp"
|
2016-09-18 13:35:54 -04:00
|
|
|
|
|
|
|
namespace WD {
|
|
|
|
|
2017-07-22 17:25:53 -04:00
|
|
|
/*!
|
|
|
|
Provides an emulation of various Western Digital drive controllers, including the
|
|
|
|
WD1770, WD1772, FDC1773 and FDC1793.
|
|
|
|
*/
|
2016-09-25 20:05:56 -04:00
|
|
|
class WD1770: public Storage::Disk::Controller {
|
2016-09-18 13:35:54 -04:00
|
|
|
public:
|
2016-11-26 23:29:30 +08:00
|
|
|
enum Personality {
|
2017-07-22 17:25:53 -04:00
|
|
|
P1770, // implies automatic motor-on management, with Type 2 commands offering a spin-up disable
|
2016-11-27 20:39:08 -08:00
|
|
|
P1772, // as per the 1770, with different stepping rates
|
|
|
|
P1773, // implements the side number-testing logic of the 1793; omits spin-up/loading logic
|
|
|
|
P1793 // implies Type 2 commands use side number testing logic; spin-up/loading is by HLD and HLT
|
2016-11-26 23:29:30 +08:00
|
|
|
};
|
2017-07-22 17:25:53 -04:00
|
|
|
|
|
|
|
/*!
|
|
|
|
Constructs an instance of the drive controller that behaves according to personality @c p.
|
|
|
|
@param p The type of controller to emulate.
|
|
|
|
*/
|
2016-11-26 23:29:30 +08:00
|
|
|
WD1770(Personality p);
|
2016-09-20 15:56:31 -04:00
|
|
|
|
2017-07-22 17:25:53 -04:00
|
|
|
/// Sets the value of the double-density input; when @c is_double_density is @c true, reads and writes double-density format data.
|
2016-09-18 13:35:54 -04:00
|
|
|
void set_is_double_density(bool is_double_density);
|
2017-07-22 17:25:53 -04:00
|
|
|
|
|
|
|
/// Writes @c value to the register at @c address. Only the low two bits of the address are decoded.
|
2016-09-18 13:35:54 -04:00
|
|
|
void set_register(int address, uint8_t value);
|
2017-07-22 17:25:53 -04:00
|
|
|
|
|
|
|
/// Fetches the value of the register @c address. Only the low two bits of the address are decoded.
|
2016-09-19 22:06:56 -04:00
|
|
|
uint8_t get_register(int address);
|
|
|
|
|
2017-07-22 17:25:53 -04:00
|
|
|
/// Runs the controller for @c number_of_cycles cycles.
|
2016-09-19 22:06:56 -04:00
|
|
|
void run_for_cycles(unsigned int number_of_cycles);
|
2016-09-20 15:56:31 -04:00
|
|
|
|
|
|
|
enum Flag: uint8_t {
|
2016-12-01 21:13:16 -05:00
|
|
|
NotReady = 0x80,
|
2016-09-20 15:56:31 -04:00
|
|
|
MotorOn = 0x80,
|
|
|
|
WriteProtect = 0x40,
|
|
|
|
RecordType = 0x20,
|
|
|
|
SpinUp = 0x20,
|
2016-12-01 21:13:16 -05:00
|
|
|
HeadLoaded = 0x20,
|
2016-09-20 15:56:31 -04:00
|
|
|
RecordNotFound = 0x10,
|
2016-09-24 22:04:54 -04:00
|
|
|
SeekError = 0x10,
|
2016-09-20 15:56:31 -04:00
|
|
|
CRCError = 0x08,
|
|
|
|
LostData = 0x04,
|
|
|
|
TrackZero = 0x04,
|
|
|
|
DataRequest = 0x02,
|
|
|
|
Index = 0x02,
|
|
|
|
Busy = 0x01
|
|
|
|
};
|
|
|
|
|
2017-07-22 17:25:53 -04:00
|
|
|
/// @returns The current value of the IRQ line output.
|
2016-12-06 21:16:29 -05:00
|
|
|
inline bool get_interrupt_request_line() { return status_.interrupt_request; }
|
2017-07-22 17:25:53 -04:00
|
|
|
|
|
|
|
/// @returns The current value of the DRQ line output.
|
2016-12-06 21:16:29 -05:00
|
|
|
inline bool get_data_request_line() { return status_.data_request; }
|
2017-07-22 17:25:53 -04:00
|
|
|
|
2016-11-21 13:21:49 +08:00
|
|
|
class Delegate {
|
|
|
|
public:
|
2016-12-01 07:41:52 -05:00
|
|
|
virtual void wd1770_did_change_output(WD1770 *wd1770) = 0;
|
2016-11-21 13:21:49 +08:00
|
|
|
};
|
|
|
|
inline void set_delegate(Delegate *delegate) { delegate_ = delegate; }
|
|
|
|
|
2016-12-01 20:12:22 -05:00
|
|
|
protected:
|
|
|
|
virtual void set_head_load_request(bool head_load);
|
|
|
|
void set_head_loaded(bool head_loaded);
|
|
|
|
|
2016-09-20 15:56:31 -04:00
|
|
|
private:
|
2016-11-26 23:29:30 +08:00
|
|
|
Personality personality_;
|
2016-12-01 07:41:52 -05:00
|
|
|
inline bool has_motor_on_line() { return (personality_ != P1793 ) && (personality_ != P1773); }
|
|
|
|
inline bool has_head_load_line() { return (personality_ == P1793 ); }
|
2016-11-26 23:29:30 +08:00
|
|
|
|
2016-11-30 22:26:02 -05:00
|
|
|
struct Status {
|
2016-11-30 22:39:55 -05:00
|
|
|
Status();
|
2016-11-30 22:26:02 -05:00
|
|
|
bool write_protect;
|
|
|
|
bool record_type;
|
|
|
|
bool spin_up;
|
|
|
|
bool record_not_found;
|
|
|
|
bool crc_error;
|
|
|
|
bool seek_error;
|
|
|
|
bool lost_data;
|
2016-12-01 07:41:52 -05:00
|
|
|
bool data_request;
|
2016-12-06 21:16:29 -05:00
|
|
|
bool interrupt_request;
|
2016-11-30 22:26:02 -05:00
|
|
|
bool busy;
|
|
|
|
enum {
|
|
|
|
One, Two, Three
|
|
|
|
} type;
|
|
|
|
} status_;
|
2016-09-24 20:12:45 -04:00
|
|
|
uint8_t track_;
|
|
|
|
uint8_t sector_;
|
|
|
|
uint8_t data_;
|
|
|
|
uint8_t command_;
|
|
|
|
|
|
|
|
int index_hole_count_;
|
2016-09-25 14:11:22 -04:00
|
|
|
int index_hole_count_target_;
|
2016-09-24 22:36:38 -04:00
|
|
|
int bits_since_token_;
|
|
|
|
int distance_into_section_;
|
2016-09-25 21:38:52 -04:00
|
|
|
bool is_awaiting_marker_value_;
|
2016-09-20 15:56:31 -04:00
|
|
|
|
2016-09-24 20:12:45 -04:00
|
|
|
int step_direction_;
|
2016-12-01 07:41:52 -05:00
|
|
|
void update_status(std::function<void(Status &)> updater);
|
2016-09-20 22:14:33 -04:00
|
|
|
|
2016-09-24 20:12:45 -04:00
|
|
|
// Tokeniser
|
2016-12-28 19:26:21 -05:00
|
|
|
enum DataMode {
|
|
|
|
Scanning,
|
|
|
|
Reading,
|
|
|
|
Writing
|
|
|
|
} data_mode_;
|
2016-09-24 20:12:45 -04:00
|
|
|
bool is_double_density_;
|
|
|
|
int shift_register_;
|
|
|
|
struct Token {
|
|
|
|
enum Type {
|
2017-01-01 20:39:19 -05:00
|
|
|
Index, ID, Data, DeletedData, Sync, Byte
|
2016-09-24 20:12:45 -04:00
|
|
|
} type;
|
|
|
|
uint8_t byte_value;
|
|
|
|
} latest_token_;
|
|
|
|
|
|
|
|
// Events
|
|
|
|
enum Event: int {
|
2016-09-25 14:11:22 -04:00
|
|
|
Command = (1 << 0), // Indicates receipt of a new command.
|
|
|
|
Token = (1 << 1), // Indicates recognition of a new token in the flux stream. Interrogate latest_token_ for details.
|
|
|
|
IndexHole = (1 << 2), // Indicates the passing of a physical index hole.
|
2016-12-01 21:13:16 -05:00
|
|
|
HeadLoad = (1 << 3), // Indicates the head has been loaded (1973 only).
|
2016-12-25 15:46:49 -05:00
|
|
|
DataWritten = (1 << 4), // Indicates that all queued bits have been written
|
2016-09-25 14:11:22 -04:00
|
|
|
|
2016-12-25 15:46:49 -05:00
|
|
|
Timer = (1 << 5), // Indicates that the delay_time_-powered timer has timed out.
|
|
|
|
IndexHoleTarget = (1 << 6) // Indicates that index_hole_count_ has reached index_hole_count_target_.
|
2016-09-24 20:12:45 -04:00
|
|
|
};
|
|
|
|
void posit_event(Event type);
|
|
|
|
int interesting_event_mask_;
|
|
|
|
int resume_point_;
|
2017-07-21 21:21:23 -04:00
|
|
|
unsigned int delay_time_;
|
2016-09-24 20:12:45 -04:00
|
|
|
|
2016-12-25 19:18:45 -05:00
|
|
|
// Output
|
|
|
|
int last_bit_;
|
|
|
|
void write_bit(int bit);
|
|
|
|
void write_byte(uint8_t byte);
|
2016-12-25 20:15:07 -05:00
|
|
|
void write_raw_short(uint16_t value);
|
2016-12-25 19:18:45 -05:00
|
|
|
|
2016-09-24 22:04:54 -04:00
|
|
|
// ID buffer
|
2016-11-21 13:21:49 +08:00
|
|
|
uint8_t header_[6];
|
|
|
|
|
2016-12-26 16:46:26 -05:00
|
|
|
// CRC generator
|
|
|
|
NumberTheory::CRC16 crc_generator_;
|
|
|
|
|
2016-12-01 20:12:22 -05:00
|
|
|
// 1793 head-loading logic
|
|
|
|
bool head_is_loaded_;
|
|
|
|
|
|
|
|
// delegate
|
2016-11-21 13:21:49 +08:00
|
|
|
Delegate *delegate_;
|
2016-09-24 22:04:54 -04:00
|
|
|
|
2016-11-21 13:21:49 +08:00
|
|
|
// Storage::Disk::Controller
|
2016-09-20 22:14:33 -04:00
|
|
|
virtual void process_input_bit(int value, unsigned int cycles_since_index_hole);
|
|
|
|
virtual void process_index_hole();
|
2016-12-25 15:46:49 -05:00
|
|
|
virtual void process_write_completed();
|
2016-09-18 13:35:54 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
2016-09-17 18:01:00 -04:00
|
|
|
|
|
|
|
#endif /* _770_hpp */
|