2016-07-10 14:17:53 +00:00
|
|
|
//
|
|
|
|
// G64.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 10/07/2016.
|
|
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "G64.hpp"
|
|
|
|
|
2017-11-10 03:04:49 +00:00
|
|
|
#include <cstring>
|
2016-07-10 22:36:52 +00:00
|
|
|
#include <vector>
|
2017-11-10 03:04:49 +00:00
|
|
|
|
2017-09-23 02:39:23 +00:00
|
|
|
#include "../../Track/PCMTrack.hpp"
|
|
|
|
#include "../../Encodings/CommodoreGCR.hpp"
|
2016-07-10 22:36:52 +00:00
|
|
|
|
2016-08-27 21:18:12 +00:00
|
|
|
using namespace Storage::Disk;
|
2016-07-10 14:17:53 +00:00
|
|
|
|
2016-11-21 12:14:09 +00:00
|
|
|
G64::G64(const char *file_name) :
|
2017-11-03 02:32:00 +00:00
|
|
|
file_(file_name) {
|
2016-07-10 14:17:53 +00:00
|
|
|
// read and check the file signature
|
2017-11-03 02:32:00 +00:00
|
|
|
if(!file_.check_signature("GCR-1541")) throw ErrorNotG64;
|
2016-07-10 14:17:53 +00:00
|
|
|
|
|
|
|
// check the version number
|
2017-11-03 02:32:00 +00:00
|
|
|
int version = file_.get8();
|
2017-03-26 18:34:47 +00:00
|
|
|
if(version != 0) throw ErrorUnknownVersion;
|
2016-07-10 14:17:53 +00:00
|
|
|
|
|
|
|
// get the number of tracks and track size
|
2017-11-03 02:32:00 +00:00
|
|
|
number_of_tracks_ = file_.get8();
|
|
|
|
maximum_track_size_ = file_.get16le();
|
2016-07-10 14:17:53 +00:00
|
|
|
}
|
|
|
|
|
2017-10-07 01:45:12 +00:00
|
|
|
int G64::get_head_position_count() {
|
2016-07-10 14:17:53 +00:00
|
|
|
// give at least 84 tracks, to yield the normal geometry but,
|
|
|
|
// if there are more, shove them in
|
2016-11-21 12:14:09 +00:00
|
|
|
return number_of_tracks_ > 84 ? number_of_tracks_ : 84;
|
2016-07-10 14:17:53 +00:00
|
|
|
}
|
|
|
|
|
2017-10-07 01:45:12 +00:00
|
|
|
std::shared_ptr<Track> G64::get_track_at_position(Track::Address address) {
|
2016-07-10 14:17:53 +00:00
|
|
|
std::shared_ptr<Track> resulting_track;
|
|
|
|
|
|
|
|
// if there's definitely no track here, return the empty track
|
|
|
|
// (TODO: should be supplying one with an index hole?)
|
2017-10-07 01:45:12 +00:00
|
|
|
if(address.position >= number_of_tracks_) return resulting_track;
|
|
|
|
if(address.head >= 1) return resulting_track;
|
2016-07-10 14:17:53 +00:00
|
|
|
|
|
|
|
// seek to this track's entry in the track table
|
2017-11-03 02:32:00 +00:00
|
|
|
file_.seek(static_cast<long>((address.position * 4) + 0xc), SEEK_SET);
|
2016-07-10 14:17:53 +00:00
|
|
|
|
|
|
|
// read the track offset
|
|
|
|
uint32_t track_offset;
|
2017-11-03 02:32:00 +00:00
|
|
|
track_offset = file_.get32le();
|
2016-07-10 14:17:53 +00:00
|
|
|
|
|
|
|
// if the track offset is zero, this track doesn't exist, so...
|
|
|
|
if(!track_offset) return resulting_track;
|
|
|
|
|
|
|
|
// seek to the track start
|
2017-11-03 02:32:00 +00:00
|
|
|
file_.seek(static_cast<long>(track_offset), SEEK_SET);
|
2016-07-10 14:17:53 +00:00
|
|
|
|
|
|
|
// get the real track length
|
|
|
|
uint16_t track_length;
|
2017-11-03 02:32:00 +00:00
|
|
|
track_length = file_.get16le();
|
2016-07-10 14:17:53 +00:00
|
|
|
|
|
|
|
// grab the byte contents of this track
|
2016-09-18 22:56:35 +00:00
|
|
|
std::vector<uint8_t> track_contents(track_length);
|
2017-11-03 02:32:00 +00:00
|
|
|
file_.read(&track_contents[0], track_length);
|
2016-07-10 14:17:53 +00:00
|
|
|
|
2016-07-10 17:42:45 +00:00
|
|
|
// seek to this track's entry in the speed zone table
|
2017-11-03 02:32:00 +00:00
|
|
|
file_.seek(static_cast<long>((address.position * 4) + 0x15c), SEEK_SET);
|
2016-07-10 14:17:53 +00:00
|
|
|
|
2016-07-10 17:42:45 +00:00
|
|
|
// read the speed zone offsrt
|
|
|
|
uint32_t speed_zone_offset;
|
2017-11-03 02:32:00 +00:00
|
|
|
speed_zone_offset = file_.get32le();
|
2016-07-10 17:42:45 +00:00
|
|
|
|
|
|
|
// if the speed zone is not constant, create a track based on the whole table; otherwise create one that's constant
|
2017-03-26 18:34:47 +00:00
|
|
|
if(speed_zone_offset > 3) {
|
2016-07-10 17:42:45 +00:00
|
|
|
// seek to start of speed zone
|
2017-11-03 02:32:00 +00:00
|
|
|
file_.seek(static_cast<long>(speed_zone_offset), SEEK_SET);
|
2016-07-10 17:42:45 +00:00
|
|
|
|
|
|
|
uint16_t speed_zone_length = (track_length + 3) >> 2;
|
|
|
|
|
|
|
|
// read the speed zone bytes
|
|
|
|
uint8_t speed_zone_contents[speed_zone_length];
|
2017-11-03 02:32:00 +00:00
|
|
|
file_.read(speed_zone_contents, speed_zone_length);
|
2016-07-10 20:10:05 +00:00
|
|
|
|
2016-07-10 22:07:53 +00:00
|
|
|
// divide track into appropriately timed PCMSegments
|
|
|
|
std::vector<PCMSegment> segments;
|
|
|
|
unsigned int current_speed = speed_zone_contents[0] >> 6;
|
|
|
|
unsigned int start_byte_in_current_speed = 0;
|
2017-03-26 18:34:47 +00:00
|
|
|
for(unsigned int byte = 0; byte < track_length; byte ++) {
|
2016-07-29 15:03:09 +00:00
|
|
|
unsigned int byte_speed = speed_zone_contents[byte >> 2] >> (6 - (byte&3)*2);
|
2017-03-26 18:34:47 +00:00
|
|
|
if(byte_speed != current_speed || byte == (track_length-1)) {
|
2016-07-10 22:07:53 +00:00
|
|
|
unsigned int number_of_bytes = byte - start_byte_in_current_speed;
|
|
|
|
|
|
|
|
PCMSegment segment;
|
2016-07-15 10:51:11 +00:00
|
|
|
segment.number_of_bits = number_of_bytes * 8;
|
2016-07-29 15:03:09 +00:00
|
|
|
segment.length_of_a_bit = Encodings::CommodoreGCR::length_of_a_bit_in_time_zone(current_speed);
|
2016-09-18 22:56:35 +00:00
|
|
|
segment.data.resize(number_of_bytes);
|
2017-11-11 20:28:40 +00:00
|
|
|
std::memcpy(&segment.data[0], &track_contents[start_byte_in_current_speed], number_of_bytes);
|
2016-07-10 22:07:53 +00:00
|
|
|
segments.push_back(std::move(segment));
|
|
|
|
|
|
|
|
current_speed = byte_speed;
|
|
|
|
start_byte_in_current_speed = byte;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
resulting_track.reset(new PCMTrack(std::move(segments)));
|
2017-03-26 18:34:47 +00:00
|
|
|
} else {
|
2016-07-10 20:10:05 +00:00
|
|
|
PCMSegment segment;
|
2016-07-15 10:51:11 +00:00
|
|
|
segment.number_of_bits = track_length * 8;
|
2017-10-21 23:49:04 +00:00
|
|
|
segment.length_of_a_bit = Encodings::CommodoreGCR::length_of_a_bit_in_time_zone(static_cast<unsigned int>(speed_zone_offset));
|
2016-07-10 20:10:05 +00:00
|
|
|
segment.data = std::move(track_contents);
|
|
|
|
|
|
|
|
resulting_track.reset(new PCMTrack(std::move(segment)));
|
2016-07-10 17:42:45 +00:00
|
|
|
}
|
2016-07-10 14:17:53 +00:00
|
|
|
|
2016-07-10 22:24:12 +00:00
|
|
|
// TODO: find out whether it's possible for a G64 to supply only a partial track. I don't think it is, which would make the
|
|
|
|
// above correct but supposing I'm wrong, the above would produce some incorrectly clocked tracks
|
|
|
|
|
2016-07-10 14:17:53 +00:00
|
|
|
return resulting_track;
|
|
|
|
}
|