2021-03-19 23:01:49 -04:00
|
|
|
//
|
|
|
|
// SpectrumTAP.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 19/03/2021.
|
|
|
|
// Copyright © 2021 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "ZXSpectrumTAP.hpp"
|
|
|
|
|
|
|
|
using namespace Storage::Tape;
|
|
|
|
|
2021-03-19 23:29:09 -04:00
|
|
|
/*
|
|
|
|
The understanding of idiomatic Spectrum data encoding below
|
|
|
|
is taken from the TZX specifications at
|
|
|
|
https://worldofspectrum.net/features/TZXformat.html ;
|
|
|
|
specifics of the TAP encoding were gained from
|
|
|
|
https://sinclair.wiki.zxnet.co.uk/wiki/TAP_format
|
|
|
|
*/
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
ZXSpectrumTAP::ZXSpectrumTAP(const std::string &file_name) : Tape(serialiser_), serialiser_(file_name) {}
|
|
|
|
|
|
|
|
ZXSpectrumTAP::Serialiser::Serialiser(const std::string &file_name) :
|
2024-12-03 09:18:05 -05:00
|
|
|
file_(file_name, FileHolder::FileMode::Read)
|
2021-03-19 23:01:49 -04:00
|
|
|
{
|
|
|
|
// Check for a continuous series of blocks through to
|
2021-03-21 20:31:09 -04:00
|
|
|
// exactly file end.
|
|
|
|
//
|
|
|
|
// To consider: could also check those blocks of type 0
|
|
|
|
// and type ff for valid checksums?
|
2024-12-03 09:18:05 -05:00
|
|
|
while(file_.tell() != file_.stats().st_size) {
|
2021-03-19 23:01:49 -04:00
|
|
|
const uint16_t block_length = file_.get16le();
|
2024-12-03 22:28:57 -05:00
|
|
|
if(file_.eof() || file_.tell() + block_length > file_.stats().st_size) {
|
2024-12-03 09:18:05 -05:00
|
|
|
throw ErrorNotZXSpectrumTAP;
|
|
|
|
}
|
2021-03-19 23:01:49 -04:00
|
|
|
|
2021-03-21 20:31:09 -04:00
|
|
|
file_.seek(block_length, SEEK_CUR);
|
2021-03-19 23:01:49 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
reset();
|
2021-03-19 23:01:49 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
bool ZXSpectrumTAP::Serialiser::is_at_end() const {
|
2021-03-28 23:25:29 -04:00
|
|
|
return file_.tell() == file_.stats().st_size && phase_ == Phase::Gap;
|
2021-03-19 23:01:49 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
void ZXSpectrumTAP::Serialiser::reset() {
|
2021-03-19 23:01:49 -04:00
|
|
|
file_.seek(0, SEEK_SET);
|
2021-03-19 23:29:09 -04:00
|
|
|
read_next_block();
|
2021-03-19 23:01:49 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:54:29 -05:00
|
|
|
Pulse ZXSpectrumTAP::Serialiser::next_pulse() {
|
2021-03-19 23:29:09 -04:00
|
|
|
// Adopt a general pattern of high then low.
|
|
|
|
Pulse pulse;
|
|
|
|
pulse.type = (distance_into_phase_ & 1) ? Pulse::Type::High : Pulse::Type::Low;
|
|
|
|
|
|
|
|
switch(phase_) {
|
|
|
|
default: break;
|
|
|
|
|
|
|
|
case Phase::PilotTone: {
|
|
|
|
// Output: pulses of length 2168;
|
|
|
|
// 8063 pulses if block type is 0, otherwise 3223;
|
|
|
|
// then a 667-length pulse followed by a 735-length pulse.
|
|
|
|
|
|
|
|
pulse.length = Time(271, 437'500); // i.e. 2168 / 3'500'000
|
|
|
|
++distance_into_phase_;
|
|
|
|
|
|
|
|
// Check whether in the last two.
|
|
|
|
if(distance_into_phase_ >= (block_type_ ? 8063 : 3223)) {
|
|
|
|
pulse.length = (distance_into_phase_ & 1) ? Time(667, 3'500'000) : Time(735, 3'500'000);
|
|
|
|
|
|
|
|
// Check whether this is the last one.
|
|
|
|
if(distance_into_phase_ == (block_type_ ? 8064 : 3224)) {
|
|
|
|
distance_into_phase_ = 0;
|
|
|
|
phase_ = Phase::Data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case Phase::Data: {
|
|
|
|
// Output two pulses of length 855 for a 0; two of length 1710 for a 1,
|
|
|
|
// from MSB to LSB.
|
|
|
|
pulse.length = (data_byte_ & 0x80) ? Time(1710, 3'500'000) : Time(855, 3'500'000);
|
|
|
|
++distance_into_phase_;
|
|
|
|
|
|
|
|
if(!(distance_into_phase_ & 1)) {
|
|
|
|
data_byte_ <<= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!(distance_into_phase_ & 15)) {
|
|
|
|
if((distance_into_phase_ >> 4) == block_length_) {
|
|
|
|
if(block_type_) {
|
|
|
|
distance_into_phase_ = 0;
|
|
|
|
phase_ = Phase::Gap;
|
|
|
|
} else {
|
|
|
|
read_next_block();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
data_byte_ = file_.get8();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case Phase::Gap:
|
|
|
|
Pulse gap;
|
|
|
|
gap.type = Pulse::Type::Zero;
|
|
|
|
gap.length = Time(1);
|
|
|
|
|
|
|
|
read_next_block();
|
|
|
|
return gap;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pulse;
|
|
|
|
}
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
void ZXSpectrumTAP::Serialiser::read_next_block() {
|
2021-03-28 23:25:29 -04:00
|
|
|
if(file_.tell() == file_.stats().st_size) {
|
2021-03-19 23:29:09 -04:00
|
|
|
phase_ = Phase::Gap;
|
|
|
|
} else {
|
|
|
|
block_length_ = file_.get16le();
|
|
|
|
data_byte_ = block_type_ = file_.get8();
|
|
|
|
phase_ = Phase::PilotTone;
|
|
|
|
}
|
|
|
|
distance_into_phase_ = 0;
|
2021-03-19 23:01:49 -04:00
|
|
|
}
|