2016-10-11 11:39:48 +00:00
|
|
|
//
|
|
|
|
// OricTAP.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 10/10/2016.
|
2018-05-13 19:19:52 +00:00
|
|
|
// Copyright 2016 Thomas Harte. All rights reserved.
|
2016-10-11 11:39:48 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "OricTAP.hpp"
|
|
|
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
using namespace Storage::Tape;
|
|
|
|
|
2024-12-04 03:28:57 +00:00
|
|
|
OricTAP::OricTAP(const std::string &file_name) : Tape(serialiser_), serialiser_(file_name) {}
|
|
|
|
|
|
|
|
|
|
|
|
OricTAP::Serialiser::Serialiser(const std::string &file_name) :
|
2024-12-03 14:21:13 +00:00
|
|
|
file_(file_name, FileHolder::FileMode::Read)
|
2016-10-11 11:39:48 +00:00
|
|
|
{
|
2021-04-29 22:00:02 +00:00
|
|
|
// Check for a sequence of at least three 0x16s followed by a 0x24.
|
|
|
|
while(true) {
|
|
|
|
const uint8_t next = file_.get8();
|
|
|
|
if(next != 0x16 && next != 0x24) {
|
|
|
|
throw ErrorNotOricTAP;
|
|
|
|
}
|
|
|
|
if(next == 0x24) {
|
|
|
|
if(file_.tell() < 4) {
|
|
|
|
throw ErrorNotOricTAP;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-10-11 11:39:48 +00:00
|
|
|
|
2024-12-04 03:28:57 +00:00
|
|
|
// Rewind and start again.
|
|
|
|
reset();
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
|
|
|
|
2024-12-04 03:28:57 +00:00
|
|
|
void OricTAP::Serialiser::reset() {
|
2017-11-03 02:32:00 +00:00
|
|
|
file_.seek(0, SEEK_SET);
|
2016-11-21 12:14:09 +00:00
|
|
|
bit_count_ = 13;
|
|
|
|
phase_ = next_phase_ = LeadIn;
|
|
|
|
phase_counter_ = 0;
|
|
|
|
pulse_counter_ = 0;
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
|
|
|
|
2024-12-04 03:28:57 +00:00
|
|
|
Pulse OricTAP::Serialiser::get_next_pulse() {
|
2016-10-11 11:39:48 +00:00
|
|
|
// Each byte byte is written as 13 bits: 0, eight bits of data, parity, three 1s.
|
2017-11-03 02:32:00 +00:00
|
|
|
if(bit_count_ == 13) {
|
|
|
|
if(next_phase_ != phase_) {
|
2016-11-21 12:14:09 +00:00
|
|
|
phase_ = next_phase_;
|
|
|
|
phase_counter_ = 0;
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
|
|
|
|
2016-11-21 12:14:09 +00:00
|
|
|
bit_count_ = 0;
|
2016-10-11 11:39:48 +00:00
|
|
|
uint8_t next_byte = 0;
|
2017-11-03 02:32:00 +00:00
|
|
|
switch(phase_) {
|
2016-10-11 11:39:48 +00:00
|
|
|
case LeadIn:
|
2016-11-21 12:14:09 +00:00
|
|
|
next_byte = phase_counter_ < 258 ? 0x16 : 0x24;
|
|
|
|
phase_counter_++;
|
2017-11-03 02:32:00 +00:00
|
|
|
if(phase_counter_ == 259) { // 256 artificial bytes plus the three in the file = 259
|
|
|
|
while(1) {
|
|
|
|
if(file_.get8() != 0x16) break;
|
2016-10-25 01:59:06 +00:00
|
|
|
}
|
2016-11-21 12:14:09 +00:00
|
|
|
next_phase_ = Header;
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Header:
|
2016-10-11 11:57:10 +00:00
|
|
|
// Counts are relative to:
|
2016-10-16 01:39:53 +00:00
|
|
|
// [0, 1]: "two bytes unused" (on the Oric 1)
|
|
|
|
// 2: program type
|
|
|
|
// 3: auto indicator
|
|
|
|
// [4, 5]: end address of data
|
|
|
|
// [6, 7]: start address of data
|
|
|
|
// 8: "unused" (on the Oric 1)
|
|
|
|
// [9...]: filename, up to NULL byte
|
2017-11-03 02:32:00 +00:00
|
|
|
next_byte = file_.get8();
|
2016-10-11 11:39:48 +00:00
|
|
|
|
2020-05-10 03:00:39 +00:00
|
|
|
if(phase_counter_ == 4) data_end_address_ = uint16_t(next_byte << 8);
|
2016-11-21 12:14:09 +00:00
|
|
|
if(phase_counter_ == 5) data_end_address_ |= next_byte;
|
2020-05-10 03:00:39 +00:00
|
|
|
if(phase_counter_ == 6) data_start_address_ = uint16_t(next_byte << 8);
|
2016-11-21 12:14:09 +00:00
|
|
|
if(phase_counter_ == 7) data_start_address_ |= next_byte;
|
2016-10-11 11:39:48 +00:00
|
|
|
|
2017-11-03 02:32:00 +00:00
|
|
|
if(phase_counter_ >= 9 && !next_byte) { // advance after the filename-ending NULL byte
|
2016-11-21 12:14:09 +00:00
|
|
|
next_phase_ = Gap;
|
2016-10-11 12:07:51 +00:00
|
|
|
}
|
2017-11-03 02:32:00 +00:00
|
|
|
if(file_.eof()) {
|
2016-11-21 12:14:09 +00:00
|
|
|
next_phase_ = End;
|
2016-11-15 04:02:03 +00:00
|
|
|
}
|
2016-11-21 12:14:09 +00:00
|
|
|
phase_counter_++;
|
2016-10-11 12:07:51 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case Gap:
|
2016-11-21 12:14:09 +00:00
|
|
|
phase_counter_++;
|
2017-11-03 02:32:00 +00:00
|
|
|
if(phase_counter_ == 8) {
|
2016-11-21 12:14:09 +00:00
|
|
|
next_phase_ = Data;
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Data:
|
2017-11-03 02:32:00 +00:00
|
|
|
next_byte = file_.get8();
|
2016-11-21 12:14:09 +00:00
|
|
|
phase_counter_++;
|
2017-11-03 02:32:00 +00:00
|
|
|
if(phase_counter_ >= (data_end_address_ - data_start_address_)+1) {
|
|
|
|
if(next_byte == 0x16) {
|
2016-11-21 12:14:09 +00:00
|
|
|
next_phase_ = LeadIn;
|
2016-10-25 01:59:06 +00:00
|
|
|
}
|
2017-11-03 02:32:00 +00:00
|
|
|
else if(file_.eof()) {
|
2016-11-21 12:14:09 +00:00
|
|
|
next_phase_ = End;
|
2016-10-25 01:59:06 +00:00
|
|
|
}
|
|
|
|
}
|
2016-10-11 11:39:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case End:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t parity = next_byte;
|
|
|
|
parity ^= (parity >> 4);
|
|
|
|
parity ^= (parity >> 2);
|
2016-10-11 12:07:51 +00:00
|
|
|
parity ^= (parity >> 1);
|
2020-05-10 03:00:39 +00:00
|
|
|
current_value_ = uint16_t((uint16_t(next_byte) << 1) | ((parity&1) << 9) | (7 << 10));
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// In slow mode, a 0 is 4 periods of 1200 Hz, a 1 is 8 periods at 2400 Hz.
|
|
|
|
// In fast mode, a 1 is a single period of 2400 Hz, a 0 is a 2400 Hz pulse followed by a 1200 Hz pulse.
|
|
|
|
// This code models fast mode.
|
2024-12-04 03:28:57 +00:00
|
|
|
Pulse pulse;
|
2016-10-11 11:39:48 +00:00
|
|
|
pulse.length.clock_rate = 4800;
|
2016-10-11 12:07:51 +00:00
|
|
|
int next_bit;
|
2016-10-11 11:39:48 +00:00
|
|
|
|
2017-11-03 02:32:00 +00:00
|
|
|
switch(phase_) {
|
2016-10-11 11:39:48 +00:00
|
|
|
case End:
|
|
|
|
pulse.type = Pulse::Zero;
|
|
|
|
pulse.length.length = 4800;
|
|
|
|
return pulse;
|
|
|
|
|
2016-10-11 12:07:51 +00:00
|
|
|
case Gap:
|
2016-11-21 12:14:09 +00:00
|
|
|
bit_count_ = 13;
|
|
|
|
pulse.type = (phase_counter_&1) ? Pulse::Low : Pulse::High;
|
2016-10-25 01:59:06 +00:00
|
|
|
pulse.length.length = 100;
|
|
|
|
return pulse;
|
2016-10-11 12:07:51 +00:00
|
|
|
|
2016-10-11 11:39:48 +00:00
|
|
|
default:
|
2016-11-21 12:14:09 +00:00
|
|
|
next_bit = current_value_ & 1;
|
2016-10-11 12:07:51 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-11-03 02:32:00 +00:00
|
|
|
if(next_bit) {
|
2016-10-11 12:07:51 +00:00
|
|
|
pulse.length.length = 1;
|
2017-11-03 02:32:00 +00:00
|
|
|
} else {
|
2016-11-21 12:14:09 +00:00
|
|
|
pulse.length.length = pulse_counter_ ? 2 : 1;
|
2016-10-11 12:07:51 +00:00
|
|
|
}
|
2016-11-21 12:14:09 +00:00
|
|
|
pulse.type = pulse_counter_ ? Pulse::High : Pulse::Low; // TODO
|
2016-10-11 12:07:51 +00:00
|
|
|
|
2016-11-21 12:14:09 +00:00
|
|
|
pulse_counter_ ^= 1;
|
2017-11-03 02:32:00 +00:00
|
|
|
if(!pulse_counter_) {
|
2016-11-21 12:14:09 +00:00
|
|
|
current_value_ >>= 1;
|
|
|
|
bit_count_++;
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
2016-10-11 12:07:51 +00:00
|
|
|
return pulse;
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|
|
|
|
|
2024-12-04 03:28:57 +00:00
|
|
|
bool OricTAP::Serialiser::is_at_end() const {
|
2016-11-21 12:14:09 +00:00
|
|
|
return phase_ == End;
|
2016-10-11 11:39:48 +00:00
|
|
|
}
|