2016-11-06 16:13:13 -05:00
|
|
|
//
|
|
|
|
// Acorn.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 06/11/2016.
|
2018-05-13 15:19:52 -04:00
|
|
|
// Copyright 2016 Thomas Harte. All rights reserved.
|
2016-11-06 16:13:13 -05:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "Acorn.hpp"
|
|
|
|
|
|
|
|
using namespace Storage::Tape::Acorn;
|
|
|
|
|
2017-07-15 19:07:35 -04:00
|
|
|
namespace {
|
|
|
|
const int PLLClockRate = 1920000;
|
|
|
|
}
|
|
|
|
|
2018-07-10 20:01:31 -04:00
|
|
|
Parser::Parser(): crc_(0x1021) {
|
2017-07-16 19:24:01 -04:00
|
|
|
shifter_.set_delegate(this);
|
|
|
|
}
|
2016-11-06 16:13:13 -05:00
|
|
|
|
2017-07-13 21:26:05 -04:00
|
|
|
int Parser::get_next_bit(const std::shared_ptr<Storage::Tape::Tape> &tape) {
|
2016-11-06 16:13:13 -05:00
|
|
|
SymbolType symbol = get_next_symbol(tape);
|
|
|
|
return (symbol == SymbolType::One) ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
2017-07-13 21:26:05 -04:00
|
|
|
int Parser::get_next_byte(const std::shared_ptr<Storage::Tape::Tape> &tape) {
|
2016-11-06 16:13:13 -05:00
|
|
|
int value = 0;
|
|
|
|
int c = 8;
|
2017-07-13 21:26:05 -04:00
|
|
|
if(get_next_bit(tape)) {
|
2016-11-06 16:13:13 -05:00
|
|
|
set_error_flag();
|
|
|
|
return -1;
|
|
|
|
}
|
2017-07-13 21:26:05 -04:00
|
|
|
while(c--) {
|
2016-11-06 16:13:13 -05:00
|
|
|
value = (value >> 1) | (get_next_bit(tape) << 7);
|
|
|
|
}
|
2017-07-13 21:26:05 -04:00
|
|
|
if(!get_next_bit(tape)) {
|
2016-11-06 16:13:13 -05:00
|
|
|
set_error_flag();
|
|
|
|
return -1;
|
|
|
|
}
|
2017-10-03 22:04:15 -04:00
|
|
|
crc_.add(static_cast<uint8_t>(value));
|
2016-11-06 16:13:13 -05:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2017-10-17 22:34:49 -04:00
|
|
|
unsigned int Parser::get_next_short(const std::shared_ptr<Storage::Tape::Tape> &tape) {
|
|
|
|
unsigned int result = static_cast<unsigned int>(get_next_byte(tape));
|
|
|
|
result |= static_cast<unsigned int>(get_next_byte(tape)) << 8;
|
2016-11-06 16:13:13 -05:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-10-17 22:34:49 -04:00
|
|
|
unsigned int Parser::get_next_word(const std::shared_ptr<Storage::Tape::Tape> &tape) {
|
|
|
|
unsigned int result = get_next_short(tape);
|
2016-11-06 16:13:13 -05:00
|
|
|
result |= get_next_short(tape) << 8;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-12-03 12:05:19 -05:00
|
|
|
void Parser::reset_crc() { crc_.reset(); }
|
|
|
|
uint16_t Parser::get_crc() { return crc_.get_value(); }
|
2016-11-06 16:13:13 -05:00
|
|
|
|
2017-07-16 19:24:01 -04:00
|
|
|
void Parser::acorn_shifter_output_bit(int value) {
|
|
|
|
push_symbol(value ? SymbolType::One : SymbolType::Zero);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Parser::process_pulse(const Storage::Tape::Tape::Pulse &pulse) {
|
|
|
|
shifter_.process_pulse(pulse);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Shifter::Shifter() :
|
|
|
|
pll_(PLLClockRate / 4800, 15),
|
|
|
|
was_high_(false),
|
|
|
|
input_pattern_(0),
|
2017-08-01 07:07:43 -04:00
|
|
|
input_bit_counter_(0),
|
|
|
|
delegate_(nullptr) {
|
2017-07-16 19:24:01 -04:00
|
|
|
pll_.set_delegate(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Shifter::process_pulse(const Storage::Tape::Tape::Pulse &pulse) {
|
2018-04-25 19:54:39 -04:00
|
|
|
pll_.run_for(Cycles(static_cast<int>(static_cast<float>(PLLClockRate) * pulse.length.get<float>())));
|
2017-07-16 19:24:01 -04:00
|
|
|
|
|
|
|
bool is_high = pulse.type == Storage::Tape::Tape::Pulse::High;
|
|
|
|
if(is_high != was_high_) {
|
|
|
|
pll_.add_pulse();
|
|
|
|
}
|
|
|
|
was_high_ = is_high;
|
|
|
|
}
|
2016-11-06 16:13:13 -05:00
|
|
|
|
2017-07-16 19:24:01 -04:00
|
|
|
void Shifter::digital_phase_locked_loop_output_bit(int value) {
|
2017-10-21 19:49:04 -04:00
|
|
|
input_pattern_ = ((input_pattern_ << 1) | static_cast<unsigned int>(value)) & 0xf;
|
2017-07-16 19:24:01 -04:00
|
|
|
switch(input_pattern_) {
|
|
|
|
case 0x5: delegate_->acorn_shifter_output_bit(0); input_pattern_ = 0; break;
|
|
|
|
case 0xf: delegate_->acorn_shifter_output_bit(1); input_pattern_ = 0; break;
|
2017-10-21 19:49:04 -04:00
|
|
|
default: break;
|
2016-11-06 16:13:13 -05:00
|
|
|
}
|
|
|
|
}
|