2017-06-04 16:37:03 -04:00
|
|
|
//
|
|
|
|
// ZX80O.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 04/06/2017.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2017 Thomas Harte. All rights reserved.
|
2017-06-04 16:37:03 -04:00
|
|
|
//
|
|
|
|
|
2017-06-11 21:38:32 -04:00
|
|
|
#include "ZX80O81P.hpp"
|
2017-06-08 21:52:13 -04:00
|
|
|
#include "../../Data/ZX8081.hpp"
|
2017-06-04 16:37:03 -04:00
|
|
|
|
|
|
|
using namespace Storage::Tape;
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
ZX80O81P::ZX80O81P(const std::string &file_name) : Tape(serialiser_), serialiser_(file_name) {}
|
|
|
|
|
|
|
|
ZX80O81P::Serialiser::Serialiser(const std::string &file_name) {
|
2024-12-03 09:21:13 -05:00
|
|
|
Storage::FileHolder file(file_name, FileHolder::FileMode::Read);
|
2017-06-04 16:37:03 -04:00
|
|
|
|
2017-06-12 19:41:59 -04:00
|
|
|
// Grab the actual file contents
|
2020-05-09 23:00:39 -04:00
|
|
|
data_.resize(size_t(file.stats().st_size));
|
|
|
|
file.read(data_.data(), size_t(file.stats().st_size));
|
2017-06-12 19:41:59 -04:00
|
|
|
|
|
|
|
// If it's a ZX81 file, prepend a file name.
|
2017-11-02 22:32:00 -04:00
|
|
|
std::string type = file.extension();
|
2017-08-27 15:19:03 -04:00
|
|
|
platform_type_ = TargetPlatform::ZX80;
|
2017-07-22 16:02:25 -04:00
|
|
|
if(type == "p" || type == "81") {
|
2017-06-12 19:41:59 -04:00
|
|
|
// TODO, maybe: prefix a proper file name; this is leaving the file nameless.
|
|
|
|
data_.insert(data_.begin(), 0x80);
|
2017-08-27 15:19:03 -04:00
|
|
|
platform_type_ = TargetPlatform::ZX81;
|
2017-06-12 19:41:59 -04:00
|
|
|
}
|
|
|
|
|
2017-11-02 22:32:00 -04:00
|
|
|
std::shared_ptr<::Storage::Data::ZX8081::File> zx_file = Storage::Data::ZX8081::FileFromData(data_);
|
|
|
|
if(!zx_file) throw ErrorNotZX80O81P;
|
2017-06-04 16:59:26 -04:00
|
|
|
|
2017-06-04 16:37:03 -04:00
|
|
|
// then rewind and start again
|
2024-12-03 22:28:57 -05:00
|
|
|
reset();
|
2017-06-04 16:37:03 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
void ZX80O81P::Serialiser::reset() {
|
2017-06-11 21:35:09 -04:00
|
|
|
data_pointer_ = 0;
|
2017-06-04 16:59:26 -04:00
|
|
|
is_past_silence_ = false;
|
2017-06-08 21:31:54 -04:00
|
|
|
has_ended_final_byte_ = false;
|
2017-06-04 16:59:26 -04:00
|
|
|
is_high_ = true;
|
2017-06-06 18:29:15 -04:00
|
|
|
bit_pointer_ = wave_pointer_ = 0;
|
2017-06-04 16:37:03 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
bool ZX80O81P::Serialiser::has_finished_data() const {
|
2017-06-11 21:35:09 -04:00
|
|
|
return (data_pointer_ == data_.size()) && !wave_pointer_ && !bit_pointer_;
|
2017-06-08 21:31:54 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:28:57 -05:00
|
|
|
bool ZX80O81P::Serialiser::is_at_end() const {
|
2017-06-08 21:31:54 -04:00
|
|
|
return has_finished_data() && has_ended_final_byte_;
|
2017-06-04 16:37:03 -04:00
|
|
|
}
|
|
|
|
|
2024-12-03 22:54:29 -05:00
|
|
|
Pulse ZX80O81P::Serialiser::next_pulse() {
|
2024-12-03 22:28:57 -05:00
|
|
|
Pulse pulse;
|
2017-06-04 16:37:03 -04:00
|
|
|
|
2017-06-04 16:59:26 -04:00
|
|
|
// Start with 1 second of silence.
|
2017-06-08 21:31:54 -04:00
|
|
|
if(!is_past_silence_ || has_finished_data()) {
|
2017-06-07 17:39:29 -04:00
|
|
|
pulse.type = Pulse::Type::Low;
|
2017-07-08 19:21:33 -04:00
|
|
|
pulse.length.length = 1;
|
2017-06-04 16:59:26 -04:00
|
|
|
pulse.length.clock_rate = 1;
|
|
|
|
is_past_silence_ = true;
|
2017-06-08 21:31:54 -04:00
|
|
|
has_ended_final_byte_ = has_finished_data();
|
2017-06-04 16:59:26 -04:00
|
|
|
return pulse;
|
|
|
|
}
|
|
|
|
|
|
|
|
// For each byte, output 8 bits and then silence.
|
2017-06-06 18:29:15 -04:00
|
|
|
if(!bit_pointer_ && !wave_pointer_) {
|
2017-06-11 21:35:09 -04:00
|
|
|
byte_ = data_[data_pointer_];
|
|
|
|
data_pointer_++;
|
2017-06-04 16:59:26 -04:00
|
|
|
bit_pointer_ = 0;
|
|
|
|
wave_pointer_ = 0;
|
|
|
|
}
|
|
|
|
|
2017-06-06 18:29:15 -04:00
|
|
|
if(!wave_pointer_) {
|
2018-05-13 15:46:14 -04:00
|
|
|
// post-waves silence (here actually a pre-waves silence) is 1300 microseconds
|
2017-06-06 18:29:15 -04:00
|
|
|
pulse.length.length = 13;
|
|
|
|
pulse.length.clock_rate = 10000;
|
2017-06-07 17:39:29 -04:00
|
|
|
pulse.type = Pulse::Type::Low;
|
2017-06-06 18:29:15 -04:00
|
|
|
|
|
|
|
wave_pointer_ ++;
|
|
|
|
} else {
|
2018-05-13 15:46:14 -04:00
|
|
|
// pulses are of length 150 microseconds
|
2017-06-04 16:59:26 -04:00
|
|
|
pulse.length.length = 3;
|
|
|
|
pulse.length.clock_rate = 20000;
|
|
|
|
|
|
|
|
if(is_high_) {
|
|
|
|
pulse.type = Pulse::Type::High;
|
|
|
|
is_high_ = false;
|
|
|
|
} else {
|
|
|
|
pulse.type = Pulse::Type::Low;
|
|
|
|
is_high_ = true;
|
|
|
|
|
2017-06-06 18:29:15 -04:00
|
|
|
// Bytes are stored MSB first.
|
2017-06-04 16:59:26 -04:00
|
|
|
int wave_count = (byte_ & (0x80 >> bit_pointer_)) ? 9 : 4;
|
|
|
|
wave_pointer_++;
|
2017-06-06 18:29:15 -04:00
|
|
|
if(wave_pointer_ == wave_count + 1) {
|
|
|
|
bit_pointer_ = (bit_pointer_ + 1)&7;
|
2017-06-04 16:59:26 -04:00
|
|
|
wave_pointer_ = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-04 16:37:03 -04:00
|
|
|
return pulse;
|
|
|
|
}
|
2017-08-27 15:19:03 -04:00
|
|
|
|
|
|
|
TargetPlatform::Type ZX80O81P::target_platform_type() {
|
2024-12-03 22:28:57 -05:00
|
|
|
return serialiser_.target_platform_type();
|
|
|
|
}
|
|
|
|
|
|
|
|
TargetPlatform::Type ZX80O81P::Serialiser::target_platform_type() {
|
2017-08-27 15:19:03 -04:00
|
|
|
return platform_type_;
|
|
|
|
}
|