2016-11-21 12:47:16 +00:00
|
|
|
//
|
|
|
|
// OricMFMDSK.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 21/11/2016.
|
|
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "OricMFMDSK.hpp"
|
2017-09-23 02:39:23 +00:00
|
|
|
|
|
|
|
#include "../../Track/PCMTrack.hpp"
|
2017-09-25 01:40:43 +00:00
|
|
|
#include "../../Encodings/MFM/Constants.hpp"
|
2017-09-28 02:14:50 +00:00
|
|
|
#include "../../Encodings/MFM/Shifter.hpp"
|
2017-09-25 01:40:43 +00:00
|
|
|
#include "../../Encodings/MFM/Encoder.hpp"
|
2017-09-28 02:14:50 +00:00
|
|
|
#include "../../Track/TrackSerialiser.hpp"
|
2016-11-21 12:47:16 +00:00
|
|
|
|
|
|
|
using namespace Storage::Disk;
|
|
|
|
|
|
|
|
OricMFMDSK::OricMFMDSK(const char *file_name) :
|
2017-03-26 18:34:47 +00:00
|
|
|
Storage::FileHolder(file_name) {
|
2016-11-21 12:47:16 +00:00
|
|
|
if(!check_signature("MFM_DISK", 8))
|
|
|
|
throw ErrorNotOricMFMDSK;
|
|
|
|
|
|
|
|
head_count_ = fgetc32le();
|
|
|
|
track_count_ = fgetc32le();
|
|
|
|
geometry_type_ = fgetc32le();
|
|
|
|
|
2016-11-26 02:13:12 +00:00
|
|
|
if(geometry_type_ < 1 || geometry_type_ > 2)
|
2016-11-21 12:47:16 +00:00
|
|
|
throw ErrorNotOricMFMDSK;
|
|
|
|
}
|
|
|
|
|
2017-03-26 18:34:47 +00:00
|
|
|
unsigned int OricMFMDSK::get_head_position_count() {
|
2016-11-21 12:47:16 +00:00
|
|
|
return track_count_;
|
|
|
|
}
|
|
|
|
|
2017-03-26 18:34:47 +00:00
|
|
|
unsigned int OricMFMDSK::get_head_count() {
|
2016-11-21 12:47:16 +00:00
|
|
|
return head_count_;
|
|
|
|
}
|
|
|
|
|
2017-03-26 18:34:47 +00:00
|
|
|
long OricMFMDSK::get_file_offset_for_position(unsigned int head, unsigned int position) {
|
2016-11-26 05:40:10 +00:00
|
|
|
long seek_offset = 0;
|
2017-03-26 18:34:47 +00:00
|
|
|
switch(geometry_type_) {
|
2016-11-26 02:13:12 +00:00
|
|
|
case 1:
|
2016-11-26 05:40:10 +00:00
|
|
|
seek_offset = (head * track_count_) + position;
|
2016-11-21 12:47:16 +00:00
|
|
|
break;
|
2016-11-26 02:13:12 +00:00
|
|
|
case 2:
|
2016-11-26 05:40:10 +00:00
|
|
|
seek_offset = (position * track_count_ * head_count_) + head;
|
2016-11-21 12:47:16 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-12-31 03:51:48 +00:00
|
|
|
return (seek_offset * 6400) + 256;
|
|
|
|
}
|
|
|
|
|
2017-09-23 00:28:11 +00:00
|
|
|
std::shared_ptr<Track> OricMFMDSK::get_track_at_position(unsigned int head, unsigned int position) {
|
2016-11-21 12:47:16 +00:00
|
|
|
PCMSegment segment;
|
2017-09-23 00:28:11 +00:00
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lock_guard(file_access_mutex_);
|
|
|
|
fseek(file_, get_file_offset_for_position(head, position), SEEK_SET);
|
|
|
|
|
|
|
|
// The file format omits clock bits. So it's not a genuine MFM capture.
|
|
|
|
// A consumer must contextually guess when an FB, FC, etc is meant to be a control mark.
|
|
|
|
size_t track_offset = 0;
|
|
|
|
uint8_t last_header[6] = {0, 0, 0, 0, 0, 0};
|
|
|
|
std::unique_ptr<Encodings::MFM::Encoder> encoder = Encodings::MFM::GetMFMEncoder(segment.data);
|
|
|
|
bool did_sync = false;
|
|
|
|
while(track_offset < 6250) {
|
2017-10-04 02:04:15 +00:00
|
|
|
uint8_t next_byte = static_cast<uint8_t>(fgetc(file_));
|
2017-09-23 00:28:11 +00:00
|
|
|
track_offset++;
|
|
|
|
|
|
|
|
switch(next_byte) {
|
|
|
|
default: {
|
|
|
|
encoder->add_byte(next_byte);
|
|
|
|
if(did_sync) {
|
|
|
|
switch(next_byte) {
|
|
|
|
default: break;
|
|
|
|
|
|
|
|
case 0xfe:
|
|
|
|
for(int byte = 0; byte < 6; byte++) {
|
2017-10-04 02:04:15 +00:00
|
|
|
last_header[byte] = static_cast<uint8_t>(fgetc(file_));
|
2017-09-23 00:28:11 +00:00
|
|
|
encoder->add_byte(last_header[byte]);
|
|
|
|
track_offset++;
|
|
|
|
if(track_offset == 6250) break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0xfb:
|
|
|
|
for(int byte = 0; byte < (128 << last_header[3]) + 2; byte++) {
|
2017-10-04 02:04:15 +00:00
|
|
|
encoder->add_byte(static_cast<uint8_t>(fgetc(file_)));
|
2017-09-23 00:28:11 +00:00
|
|
|
track_offset++;
|
|
|
|
if(track_offset == 6250) break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-12-28 23:50:28 +00:00
|
|
|
}
|
|
|
|
|
2017-09-23 00:28:11 +00:00
|
|
|
did_sync = false;
|
|
|
|
}
|
|
|
|
break;
|
2016-11-26 05:40:10 +00:00
|
|
|
|
2017-09-23 00:28:11 +00:00
|
|
|
case 0xa1: // a synchronisation mark that implies a sector or header coming
|
|
|
|
encoder->output_short(Storage::Encodings::MFM::MFMSync);
|
|
|
|
did_sync = true;
|
|
|
|
break;
|
2016-11-26 05:40:10 +00:00
|
|
|
|
2017-09-23 00:28:11 +00:00
|
|
|
case 0xc2: // an 'ordinary' synchronisation mark
|
|
|
|
encoder->output_short(Storage::Encodings::MFM::MFMIndexSync);
|
|
|
|
break;
|
|
|
|
}
|
2016-11-26 05:40:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
segment.number_of_bits = (unsigned int)(segment.data.size() * 8);
|
2016-11-21 12:47:16 +00:00
|
|
|
|
|
|
|
std::shared_ptr<PCMTrack> track(new PCMTrack(segment));
|
|
|
|
return track;
|
|
|
|
}
|
2016-12-31 03:51:48 +00:00
|
|
|
|
2017-09-23 00:28:11 +00:00
|
|
|
void OricMFMDSK::set_track_at_position(unsigned int head, unsigned int position, const std::shared_ptr<Track> &track) {
|
2017-09-28 02:14:50 +00:00
|
|
|
PCMSegment segment = Storage::Disk::track_serialisation(*track, Storage::Encodings::MFM::MFMBitLength);
|
|
|
|
Storage::Encodings::MFM::Shifter shifter;
|
|
|
|
shifter.set_is_double_density(true);
|
|
|
|
shifter.set_should_obey_syncs(true);
|
|
|
|
std::vector<uint8_t> parsed_track;
|
|
|
|
int size = 0;
|
|
|
|
int offset = 0;
|
|
|
|
bool capture_size = false;
|
|
|
|
|
|
|
|
for(unsigned int bit = 0; bit < segment.number_of_bits; ++bit) {
|
|
|
|
shifter.add_input_bit(segment.bit(bit));
|
|
|
|
if(shifter.get_token() == Storage::Encodings::MFM::Shifter::Token::None) continue;
|
|
|
|
parsed_track.push_back(shifter.get_byte());
|
|
|
|
|
|
|
|
if(offset) {
|
|
|
|
offset--;
|
|
|
|
if(!offset) {
|
|
|
|
shifter.set_should_obey_syncs(true);
|
|
|
|
}
|
|
|
|
if(capture_size && offset == 2) {
|
|
|
|
size = parsed_track.back();
|
|
|
|
capture_size = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( shifter.get_token() == Storage::Encodings::MFM::Shifter::Token::Data ||
|
|
|
|
shifter.get_token() == Storage::Encodings::MFM::Shifter::Token::DeletedData) {
|
|
|
|
offset = 128 << size;
|
|
|
|
shifter.set_should_obey_syncs(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(shifter.get_token() == Storage::Encodings::MFM::Shifter::Token::ID) {
|
|
|
|
offset = 6;
|
|
|
|
shifter.set_should_obey_syncs(false);
|
|
|
|
capture_size = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-31 04:12:46 +00:00
|
|
|
long file_offset = get_file_offset_for_position(head, position);
|
2016-12-31 16:48:46 +00:00
|
|
|
|
2017-09-28 02:14:50 +00:00
|
|
|
std::lock_guard<std::mutex> lock_guard(file_access_mutex_);
|
|
|
|
fseek(file_, file_offset, SEEK_SET);
|
2016-12-31 17:51:52 +00:00
|
|
|
size_t track_size = std::min((size_t)6400, parsed_track.size());
|
|
|
|
fwrite(parsed_track.data(), 1, track_size, file_);
|
2016-12-31 03:51:48 +00:00
|
|
|
}
|