2016-11-06 16:13:13 -05:00
|
|
|
//
|
|
|
|
// Acorn.hpp
|
|
|
|
// 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
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef Storage_Tape_Parsers_Acorn_hpp
|
|
|
|
#define Storage_Tape_Parsers_Acorn_hpp
|
|
|
|
|
|
|
|
#include "TapeParser.hpp"
|
2020-01-19 23:14:35 -05:00
|
|
|
#include "../../../Numeric/CRC.hpp"
|
2017-09-22 22:39:23 -04:00
|
|
|
#include "../../Disk/DPLL/DigitalPhaseLockedLoop.hpp"
|
2016-11-06 16:13:13 -05:00
|
|
|
|
|
|
|
namespace Storage {
|
|
|
|
namespace Tape {
|
|
|
|
namespace Acorn {
|
|
|
|
|
2020-01-12 17:25:21 -05:00
|
|
|
class Shifter {
|
2017-07-16 19:24:01 -04:00
|
|
|
public:
|
|
|
|
Shifter();
|
|
|
|
|
|
|
|
void process_pulse(const Storage::Tape::Tape::Pulse &pulse);
|
|
|
|
|
|
|
|
class Delegate {
|
|
|
|
public:
|
|
|
|
virtual void acorn_shifter_output_bit(int value) = 0;
|
|
|
|
};
|
|
|
|
void set_delegate(Delegate *delegate) {
|
|
|
|
delegate_ = delegate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void digital_phase_locked_loop_output_bit(int value);
|
|
|
|
|
|
|
|
private:
|
2020-01-12 17:25:21 -05:00
|
|
|
Storage::DigitalPhaseLockedLoop<Shifter, 15> pll_;
|
2017-07-16 19:24:01 -04:00
|
|
|
bool was_high_;
|
|
|
|
|
|
|
|
unsigned int input_pattern_;
|
|
|
|
unsigned int input_bit_counter_;
|
|
|
|
|
|
|
|
Delegate *delegate_;
|
|
|
|
};
|
|
|
|
|
2016-11-06 16:13:13 -05:00
|
|
|
enum class SymbolType {
|
|
|
|
One, Zero
|
|
|
|
};
|
|
|
|
|
2017-07-16 19:24:01 -04:00
|
|
|
class Parser: public Storage::Tape::Parser<SymbolType>, public Shifter::Delegate {
|
2016-11-06 16:13:13 -05:00
|
|
|
public:
|
|
|
|
Parser();
|
|
|
|
|
|
|
|
int get_next_bit(const std::shared_ptr<Storage::Tape::Tape> &tape);
|
|
|
|
int get_next_byte(const std::shared_ptr<Storage::Tape::Tape> &tape);
|
2017-10-17 22:34:49 -04:00
|
|
|
unsigned int get_next_short(const std::shared_ptr<Storage::Tape::Tape> &tape);
|
|
|
|
unsigned int get_next_word(const std::shared_ptr<Storage::Tape::Tape> &tape);
|
2016-11-06 16:13:13 -05:00
|
|
|
void reset_crc();
|
|
|
|
uint16_t get_crc();
|
|
|
|
|
2017-07-16 19:24:01 -04:00
|
|
|
void acorn_shifter_output_bit(int value);
|
|
|
|
void process_pulse(const Storage::Tape::Tape::Pulse &pulse);
|
2017-07-15 19:07:35 -04:00
|
|
|
|
2016-11-06 16:13:13 -05:00
|
|
|
private:
|
2017-07-16 19:24:01 -04:00
|
|
|
bool did_update_shifter(int new_value, int length);
|
2018-07-10 20:01:31 -04:00
|
|
|
CRC::Generator<uint16_t, 0x0000, 0x0000, false, false> crc_;
|
2017-07-16 19:24:01 -04:00
|
|
|
Shifter shifter_;
|
2016-11-06 16:13:13 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* Acorn_hpp */
|