2016-09-18 17:35:54 +00:00
|
|
|
//
|
|
|
|
// MFM.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 18/09/2016.
|
|
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef Storage_Disk_Encodings_MFM_hpp
|
|
|
|
#define Storage_Disk_Encodings_MFM_hpp
|
|
|
|
|
|
|
|
#include <cstdint>
|
2016-09-18 20:53:21 +00:00
|
|
|
#include <vector>
|
2017-09-25 00:07:56 +00:00
|
|
|
#include "Constants.hpp"
|
|
|
|
#include "../../Disk.hpp"
|
|
|
|
#include "../../Controller/DiskController.hpp"
|
|
|
|
#include "../../../../NumberTheory/CRC.hpp"
|
2016-09-18 17:35:54 +00:00
|
|
|
|
|
|
|
namespace Storage {
|
|
|
|
namespace Encodings {
|
2016-09-18 20:53:21 +00:00
|
|
|
namespace MFM {
|
2016-09-18 17:35:54 +00:00
|
|
|
|
2017-08-11 18:24:50 +00:00
|
|
|
/*!
|
|
|
|
Represents a single [M]FM sector, identified by its track, side and sector records, a blob of data
|
|
|
|
and a few extra flags of metadata.
|
|
|
|
*/
|
2016-09-18 20:53:21 +00:00
|
|
|
struct Sector {
|
2017-08-11 18:24:50 +00:00
|
|
|
uint8_t track, side, sector, size;
|
2016-09-18 20:53:21 +00:00
|
|
|
std::vector<uint8_t> data;
|
2017-08-11 18:24:50 +00:00
|
|
|
|
|
|
|
bool has_data_crc_error;
|
|
|
|
bool has_header_crc_error;
|
|
|
|
bool is_deleted;
|
|
|
|
|
|
|
|
Sector() : track(0), side(0), sector(0), size(0), has_data_crc_error(false), has_header_crc_error(false), is_deleted(false) {}
|
2016-09-18 20:53:21 +00:00
|
|
|
};
|
|
|
|
|
2017-08-11 18:24:50 +00:00
|
|
|
extern const size_t DefaultSectorGapLength;
|
2017-08-11 18:27:07 +00:00
|
|
|
/*!
|
|
|
|
Converts a vector of sectors into a properly-encoded MFM track.
|
|
|
|
|
|
|
|
@param sectors The sectors to write.
|
|
|
|
@param sector_gap_length If specified, sets the distance in whole bytes between each ID and its data.
|
|
|
|
@param sector_gap_filler_byte If specified, sets the value (unencoded) that is used to populate the gap between each ID and its data.
|
|
|
|
*/
|
2017-08-11 18:24:50 +00:00
|
|
|
std::shared_ptr<Storage::Disk::Track> GetMFMTrackWithSectors(const std::vector<Sector> §ors, size_t sector_gap_length = DefaultSectorGapLength, uint8_t sector_gap_filler_byte = 0x4e);
|
2017-08-11 18:27:07 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
Converts a vector of sectors into a properly-encoded FM track.
|
|
|
|
|
|
|
|
@param sectors The sectors to write.
|
|
|
|
@param sector_gap_length If specified, sets the distance in whole bytes between each ID and its data.
|
|
|
|
@param sector_gap_filler_byte If specified, sets the value (unencoded) that is used to populate the gap between each ID and its data.
|
|
|
|
*/
|
2017-08-11 18:24:50 +00:00
|
|
|
std::shared_ptr<Storage::Disk::Track> GetFMTrackWithSectors(const std::vector<Sector> §ors, size_t sector_gap_length = DefaultSectorGapLength, uint8_t sector_gap_filler_byte = 0x4e);
|
2016-09-18 20:53:21 +00:00
|
|
|
|
2016-11-26 05:39:20 +00:00
|
|
|
class Encoder {
|
|
|
|
public:
|
|
|
|
Encoder(std::vector<uint8_t> &target);
|
|
|
|
virtual void add_byte(uint8_t input) = 0;
|
|
|
|
virtual void add_index_address_mark() = 0;
|
|
|
|
virtual void add_ID_address_mark() = 0;
|
|
|
|
virtual void add_data_address_mark() = 0;
|
|
|
|
virtual void add_deleted_data_address_mark() = 0;
|
2016-12-28 23:50:28 +00:00
|
|
|
virtual void output_short(uint16_t value);
|
2017-08-11 18:24:50 +00:00
|
|
|
|
|
|
|
/// Outputs the CRC for all data since the last address mask; if @c incorrectly is @c true then outputs an incorrect CRC.
|
|
|
|
void add_crc(bool incorrectly);
|
2016-11-26 05:39:20 +00:00
|
|
|
|
|
|
|
protected:
|
2016-12-28 23:29:37 +00:00
|
|
|
NumberTheory::CRC16 crc_generator_;
|
2016-11-26 05:39:20 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<uint8_t> &target_;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::unique_ptr<Encoder> GetMFMEncoder(std::vector<uint8_t> &target);
|
|
|
|
std::unique_ptr<Encoder> GetFMEncoder(std::vector<uint8_t> &target);
|
|
|
|
|
2016-12-26 01:00:57 +00:00
|
|
|
class Parser: public Storage::Disk::Controller {
|
|
|
|
public:
|
|
|
|
Parser(bool is_mfm, const std::shared_ptr<Storage::Disk::Disk> &disk);
|
2016-12-26 19:24:33 +00:00
|
|
|
Parser(bool is_mfm, const std::shared_ptr<Storage::Disk::Track> &track);
|
2016-12-26 01:00:57 +00:00
|
|
|
|
|
|
|
/*!
|
|
|
|
Attempts to read the sector located at @c track and @c sector.
|
|
|
|
|
|
|
|
@returns a sector if one was found; @c nullptr otherwise.
|
|
|
|
*/
|
2017-08-11 14:29:13 +00:00
|
|
|
std::shared_ptr<Storage::Encodings::MFM::Sector> get_sector(uint8_t head, uint8_t track, uint8_t sector);
|
2016-12-26 01:00:57 +00:00
|
|
|
|
2016-12-31 00:59:23 +00:00
|
|
|
/*!
|
|
|
|
Attempts to read the track at @c track, starting from the index hole.
|
|
|
|
|
2016-12-31 04:10:52 +00:00
|
|
|
Decodes data bits only; clocks are omitted. Synchronisation values begin a new
|
|
|
|
byte. If a synchronisation value begins partway through a byte then
|
|
|
|
synchronisation-contributing bits will appear both in the preceding byte and
|
|
|
|
in the next.
|
2016-12-31 00:59:23 +00:00
|
|
|
|
2016-12-31 04:10:52 +00:00
|
|
|
@returns a vector of data found.
|
2016-12-31 00:59:23 +00:00
|
|
|
*/
|
2016-12-31 04:10:52 +00:00
|
|
|
std::vector<uint8_t> get_track(uint8_t track);
|
2016-12-31 00:59:23 +00:00
|
|
|
|
2016-12-26 01:00:57 +00:00
|
|
|
private:
|
2016-12-26 19:24:33 +00:00
|
|
|
Parser(bool is_mfm);
|
|
|
|
|
2017-08-11 14:29:13 +00:00
|
|
|
std::shared_ptr<Storage::Disk::Drive> drive_;
|
2016-12-26 01:00:57 +00:00
|
|
|
unsigned int shift_register_;
|
|
|
|
int index_count_;
|
2017-08-11 14:29:13 +00:00
|
|
|
uint8_t track_, head_;
|
2016-12-26 01:00:57 +00:00
|
|
|
int bit_count_;
|
|
|
|
NumberTheory::CRC16 crc_generator_;
|
|
|
|
bool is_mfm_;
|
|
|
|
|
2016-12-31 00:59:23 +00:00
|
|
|
void seek_to_track(uint8_t track);
|
2017-09-10 23:23:23 +00:00
|
|
|
void process_input_bit(int value);
|
2016-12-26 01:00:57 +00:00
|
|
|
void process_index_hole();
|
|
|
|
uint8_t get_next_byte();
|
2016-12-31 00:59:23 +00:00
|
|
|
|
2016-12-31 05:11:31 +00:00
|
|
|
uint8_t get_byte_for_shift_value(uint16_t value);
|
|
|
|
|
2016-12-26 01:00:57 +00:00
|
|
|
std::shared_ptr<Storage::Encodings::MFM::Sector> get_next_sector();
|
|
|
|
std::shared_ptr<Storage::Encodings::MFM::Sector> get_sector(uint8_t sector);
|
2016-12-31 04:10:52 +00:00
|
|
|
std::vector<uint8_t> get_track();
|
2017-08-11 14:29:13 +00:00
|
|
|
|
|
|
|
std::map<int, std::shared_ptr<Storage::Encodings::MFM::Sector>> sectors_by_index_;
|
2017-08-11 22:40:16 +00:00
|
|
|
std::set<int> decoded_tracks_;
|
2017-08-11 14:29:13 +00:00
|
|
|
int get_index(uint8_t head, uint8_t track, uint8_t sector);
|
2016-12-26 01:00:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2016-09-18 20:53:21 +00:00
|
|
|
}
|
2016-09-18 17:35:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* MFM_hpp */
|