2019-10-03 22:10:10 -04:00
|
|
|
//
|
|
|
|
// MSA.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 03/10/2019.
|
|
|
|
// Copyright © 2019 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "MSA.hpp"
|
|
|
|
|
2019-10-03 22:41:20 -04:00
|
|
|
#include "Utility/ImplicitSectors.hpp"
|
|
|
|
|
2019-12-19 22:27:50 -05:00
|
|
|
#include <cassert>
|
|
|
|
|
2019-10-03 22:10:10 -04:00
|
|
|
using namespace Storage::Disk;
|
|
|
|
|
|
|
|
MSA::MSA(const std::string &file_name) :
|
|
|
|
file_(file_name) {
|
2019-10-03 22:41:20 -04:00
|
|
|
const auto signature = file_.get16be();
|
|
|
|
if(signature != 0x0e0f) throw Error::InvalidFormat;
|
|
|
|
|
|
|
|
sectors_per_track_ = file_.get16be();
|
|
|
|
sides_ = 1 + file_.get16be();
|
|
|
|
starting_track_ = file_.get16be();
|
|
|
|
ending_track_ = file_.get16be();
|
|
|
|
|
|
|
|
// Create the uncompressed track list.
|
|
|
|
while(true) {
|
|
|
|
const auto data_length = file_.get16be();
|
|
|
|
if(file_.eof()) break;
|
|
|
|
|
|
|
|
if(data_length == sectors_per_track_ * 512) {
|
|
|
|
// This is an uncompressed track.
|
|
|
|
uncompressed_tracks_.push_back(file_.read(data_length));
|
|
|
|
} else {
|
2019-12-19 22:27:50 -05:00
|
|
|
#ifndef NDEBUG
|
|
|
|
const auto start_of_track = file_.tell();
|
|
|
|
#endif
|
2019-10-03 22:41:20 -04:00
|
|
|
// This is an RLE-compressed track.
|
|
|
|
std::vector<uint8_t> track;
|
|
|
|
track.reserve(sectors_per_track_ * 512);
|
|
|
|
uint16_t pointer = 0;
|
|
|
|
while(pointer < data_length) {
|
|
|
|
const auto byte = file_.get8();
|
|
|
|
|
|
|
|
// Compression scheme: if the byte E5 is encountered, an RLE run follows.
|
|
|
|
// An RLE run is encoded as the byte to repeat plus a 16-bit repeat count.
|
|
|
|
if(byte != 0xe5) {
|
|
|
|
track.push_back(byte);
|
|
|
|
++pointer;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
pointer += 4;
|
|
|
|
if(pointer > data_length) break;
|
|
|
|
|
|
|
|
const auto value = file_.get8();
|
|
|
|
auto count = file_.get16be();
|
|
|
|
while(count--) {
|
|
|
|
track.push_back(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-19 22:27:50 -05:00
|
|
|
#ifndef NDEBUG
|
|
|
|
assert(file_.tell() - start_of_track == pointer);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if(pointer != data_length || track.size() != sectors_per_track_ * 512)
|
|
|
|
throw Error::InvalidFormat;
|
2019-10-03 22:41:20 -04:00
|
|
|
uncompressed_tracks_.push_back(std::move(track));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-19 22:27:50 -05:00
|
|
|
if(uncompressed_tracks_.size() != size_t((ending_track_ - starting_track_ + 1)*sides_))
|
|
|
|
throw Error::InvalidFormat;
|
2019-10-03 22:10:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<::Storage::Disk::Track> MSA::get_track_at_position(::Storage::Disk::Track::Address address) {
|
2019-10-03 22:41:20 -04:00
|
|
|
if(address.head >= sides_) return nullptr;
|
|
|
|
|
|
|
|
const auto position = address.position.as_int();
|
|
|
|
if(position < starting_track_) return nullptr;
|
2019-12-24 23:01:52 -05:00
|
|
|
if(position > ending_track_) return nullptr;
|
2019-10-03 22:41:20 -04:00
|
|
|
|
2019-12-19 22:27:50 -05:00
|
|
|
const auto &track = uncompressed_tracks_[size_t(position - starting_track_) * size_t(sides_) + size_t(address.head)];
|
|
|
|
assert(!track.empty());
|
2019-11-03 21:18:29 -05:00
|
|
|
return track_for_sectors(track.data(), sectors_per_track_, uint8_t(position), uint8_t(address.head), 1, 2, true);
|
2019-10-03 22:10:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
HeadPosition MSA::get_maximum_head_position() {
|
2019-12-24 23:01:52 -05:00
|
|
|
return HeadPosition(ending_track_ + 1);
|
2019-10-03 22:41:20 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int MSA::get_head_count() {
|
|
|
|
return sides_;
|
2019-10-03 22:10:10 -04:00
|
|
|
}
|