mirror of
https://github.com/TomHarte/CLK.git
synced 2024-11-18 17:06:15 +00:00
84 lines
1.6 KiB
C++
84 lines
1.6 KiB
C++
//
|
|
// DMA.hpp
|
|
// Clock Signal
|
|
//
|
|
// Created by Thomas Harte on 21/11/2023.
|
|
// Copyright © 2023 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#ifndef DMA_hpp
|
|
#define DMA_hpp
|
|
|
|
#include "../../Numeric/RegisterSizes.hpp"
|
|
|
|
namespace PCCompatible {
|
|
|
|
class DMA {
|
|
public:
|
|
void flip_flop_reset() {
|
|
next_access_low = true;
|
|
}
|
|
|
|
void mask_reset() {
|
|
// TODO: set all mask bits off.
|
|
}
|
|
|
|
void master_reset() {
|
|
flip_flop_reset();
|
|
// TODO: clear status, set all mask bits on.
|
|
}
|
|
|
|
template <int address>
|
|
void write(uint8_t value) {
|
|
constexpr int channel = (address >> 1) & 3;
|
|
constexpr bool is_count = address & 1;
|
|
|
|
next_access_low ^= true;
|
|
if(next_access_low) {
|
|
if constexpr (is_count) {
|
|
channels_[channel].count.halves.high = value;
|
|
} else {
|
|
channels_[channel].address.halves.high = value;
|
|
}
|
|
} else {
|
|
if constexpr (is_count) {
|
|
channels_[channel].count.halves.low = value;
|
|
} else {
|
|
channels_[channel].address.halves.low = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
template <int address>
|
|
uint8_t read() {
|
|
constexpr int channel = (address >> 1) & 3;
|
|
constexpr bool is_count = address & 1;
|
|
|
|
next_access_low ^= true;
|
|
if(next_access_low) {
|
|
if constexpr (is_count) {
|
|
return channels_[channel].count.halves.high;
|
|
} else {
|
|
return channels_[channel].address.halves.high;
|
|
}
|
|
} else {
|
|
if constexpr (is_count) {
|
|
return channels_[channel].count.halves.low;
|
|
} else {
|
|
return channels_[channel].address.halves.low;
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
bool next_access_low = true;
|
|
|
|
struct Channel {
|
|
CPU::RegisterPair16 address, count;
|
|
} channels_[4];
|
|
};
|
|
|
|
}
|
|
|
|
#endif /* DMA_hpp */
|