mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-26 15:32:04 +00:00
260 lines
6.4 KiB
C++
260 lines
6.4 KiB
C++
//
|
|
// PIT.hpp
|
|
// Clock Signal
|
|
//
|
|
// Created by Thomas Harte on 20/11/2023.
|
|
// Copyright © 2023 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#ifndef PIT_hpp
|
|
#define PIT_hpp
|
|
|
|
namespace PCCompatible {
|
|
|
|
template <bool is_8254, typename PITObserver>
|
|
class i8253 {
|
|
public:
|
|
i8253(PITObserver &observer) : observer_(observer) {}
|
|
|
|
template <int channel> uint8_t read() {
|
|
return channels_[channel].read();
|
|
}
|
|
|
|
template <int channel> void write(uint8_t value) {
|
|
channels_[channel].template write<channel>(observer_, value);
|
|
}
|
|
|
|
void set_mode(uint8_t value) {
|
|
const int channel_id = (value >> 6) & 3;
|
|
if(channel_id == 3) {
|
|
// TODO: decode rest of read-back command.
|
|
read_back_ = is_8254;
|
|
return;
|
|
}
|
|
switch(channel_id) {
|
|
case 0: channels_[0].template set_mode<0>(observer_, value); break;
|
|
case 1: channels_[1].template set_mode<1>(observer_, value); break;
|
|
case 2: channels_[2].template set_mode<2>(observer_, value); break;
|
|
}
|
|
}
|
|
|
|
void run_for(Cycles cycles) {
|
|
// TODO: be intelligent enough to take ticks outside the loop when appropriate.
|
|
auto ticks = cycles.as<int>();
|
|
while(ticks--) {
|
|
channels_[0].template advance<0>(observer_, 1);
|
|
channels_[1].template advance<1>(observer_, 1);
|
|
channels_[2].template advance<2>(observer_, 1);
|
|
}
|
|
}
|
|
|
|
private:
|
|
// The target for output changes.
|
|
PITObserver &observer_;
|
|
|
|
// Supported only on 8254s.
|
|
bool read_back_ = false;
|
|
|
|
enum class LatchMode {
|
|
LowOnly,
|
|
HighOnly,
|
|
LowHigh,
|
|
};
|
|
|
|
enum class OperatingMode {
|
|
InterruptOnTerminalCount = 0,
|
|
HardwareRetriggerableOneShot = 1,
|
|
RateGenerator = 2,
|
|
SquareWaveGenerator = 3,
|
|
SoftwareTriggeredStrobe = 4,
|
|
HardwareTriggeredStrobe = 5,
|
|
};
|
|
|
|
struct Channel {
|
|
LatchMode latch_mode = LatchMode::LowHigh;
|
|
OperatingMode mode = OperatingMode::InterruptOnTerminalCount;
|
|
bool is_bcd = false;
|
|
|
|
bool gated = false;
|
|
bool awaiting_reload = true;
|
|
|
|
uint16_t counter = 0;
|
|
uint16_t reload = 0;
|
|
uint16_t latch = 0;
|
|
bool output = false;
|
|
|
|
bool next_access_high = false;
|
|
|
|
void latch_value() {
|
|
latch = counter;
|
|
}
|
|
|
|
template <int channel>
|
|
void set_mode(PITObserver &observer, uint8_t value) {
|
|
switch((value >> 4) & 3) {
|
|
default:
|
|
latch_value();
|
|
return;
|
|
|
|
case 1: latch_mode = LatchMode::LowOnly; break;
|
|
case 2: latch_mode = LatchMode::HighOnly; break;
|
|
case 3: latch_mode = LatchMode::LowHigh; break;
|
|
}
|
|
is_bcd = value & 1;
|
|
next_access_high = false;
|
|
|
|
const auto operating_mode = (value >> 1) & 7;
|
|
switch(operating_mode) {
|
|
default: mode = OperatingMode(operating_mode); break;
|
|
case 6: mode = OperatingMode::RateGenerator; break;
|
|
case 7: mode = OperatingMode::SquareWaveGenerator; break;
|
|
}
|
|
|
|
// Set up operating mode.
|
|
switch(mode) {
|
|
default:
|
|
printf("PIT: unimplemented mode %d\n", int(mode));
|
|
break;
|
|
|
|
case OperatingMode::InterruptOnTerminalCount:
|
|
case OperatingMode::HardwareRetriggerableOneShot:
|
|
set_output<channel>(observer, false);
|
|
awaiting_reload = true;
|
|
break;
|
|
|
|
case OperatingMode::RateGenerator:
|
|
case OperatingMode::SquareWaveGenerator:
|
|
set_output<channel>(observer, true);
|
|
awaiting_reload = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
template <int channel>
|
|
void advance(PITObserver &observer, int ticks) {
|
|
if(gated || awaiting_reload) return;
|
|
|
|
// TODO: BCD mode is completely ignored below. Possibly not too important.
|
|
switch(mode) {
|
|
case OperatingMode::InterruptOnTerminalCount:
|
|
case OperatingMode::HardwareRetriggerableOneShot:
|
|
// Output goes permanently high upon a tick from 1 to 0; reload value is not reused.
|
|
set_output<channel>(observer, output | (counter <= ticks));
|
|
counter -= ticks;
|
|
break;
|
|
|
|
case OperatingMode::SquareWaveGenerator: {
|
|
ticks <<= 1;
|
|
do {
|
|
// If there's a step from 1 to 0 within the next batch of ticks,
|
|
// toggle output and apply a reload.
|
|
if(counter && ticks >= counter) {
|
|
set_output<channel>(observer, output ^ true);
|
|
ticks -= counter;
|
|
|
|
const uint16_t reload_mask = output ? 0xffff : 0xfffe;
|
|
counter = reload & reload_mask;
|
|
|
|
continue;
|
|
}
|
|
counter -= ticks;
|
|
} while(false);
|
|
} break;
|
|
|
|
case OperatingMode::RateGenerator:
|
|
do {
|
|
// Check for a step from 2 to 1 within the next batch of ticks, which would cause output
|
|
// to go high.
|
|
if(counter > 1 && ticks >= counter - 1) {
|
|
set_output<channel>(observer, true);
|
|
ticks -= counter - 1;
|
|
counter = 1;
|
|
continue;
|
|
}
|
|
|
|
// If there is a step from 1 to 0, reload and set output back to low.
|
|
if(counter && ticks >= counter) {
|
|
set_output<channel>(observer, false);
|
|
ticks -= counter;
|
|
counter = reload;
|
|
continue;
|
|
}
|
|
|
|
// Otherwise, just continue.
|
|
counter -= ticks;
|
|
} while(false);
|
|
break;
|
|
|
|
default:
|
|
// TODO.
|
|
break;
|
|
}
|
|
}
|
|
|
|
template <int channel>
|
|
void write([[maybe_unused]] PITObserver &observer, uint8_t value) {
|
|
switch(latch_mode) {
|
|
case LatchMode::LowOnly:
|
|
reload = (reload & 0xff00) | value;
|
|
break;
|
|
case LatchMode::HighOnly:
|
|
reload = uint16_t((reload & 0x00ff) | (value << 8));
|
|
break;
|
|
case LatchMode::LowHigh:
|
|
next_access_high ^= true;
|
|
if(next_access_high) {
|
|
reload = (reload & 0xff00) | value;
|
|
awaiting_reload = true;
|
|
return;
|
|
}
|
|
|
|
reload = uint16_t((reload & 0x00ff) | (value << 8));
|
|
break;
|
|
}
|
|
|
|
awaiting_reload = false;
|
|
|
|
switch(mode) {
|
|
default:
|
|
counter = reload;
|
|
break;
|
|
|
|
case OperatingMode::SquareWaveGenerator:
|
|
counter = reload & ~1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
uint8_t read() {
|
|
switch(latch_mode) {
|
|
case LatchMode::LowOnly: return uint8_t(latch);
|
|
case LatchMode::HighOnly: return uint8_t(latch >> 8);
|
|
default:
|
|
case LatchMode::LowHigh:
|
|
next_access_high ^= true;
|
|
return next_access_high ? uint8_t(latch) : uint8_t(latch >> 8);
|
|
break;
|
|
}
|
|
}
|
|
|
|
template <int channel>
|
|
void set_output(PITObserver &observer, bool level) {
|
|
if(output == level) {
|
|
return;
|
|
}
|
|
|
|
// TODO: how should time be notified?
|
|
observer.template update_output<channel>(level);
|
|
output = level;
|
|
}
|
|
} channels_[3];
|
|
|
|
// TODO:
|
|
//
|
|
// RateGenerator: output goes high if gated.
|
|
};
|
|
|
|
}
|
|
|
|
#endif /* PIT_hpp */
|