1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
use crate::memory::{ActiveInterrupt, Memory, Port};

// MOS 6520

/// The registers associated with a single port in a MOS 6520 PIA.
struct PiaPortRegisters {
  /// The port itself.
  port: Box<dyn Port>,

  /// If the DDR is write, the current written value.
  writes: u8,

  /// Data direction register. Each bit controls whether the line is an input (0) or output (1)
  ddr: u8,

  // Control register. Each bit has a specific function.
  pub control: u8,
}

impl PiaPortRegisters {
  /// Create a new PortRegisters with the given port.
  pub fn new(port: Box<dyn Port>) -> Self {
    Self {
      port,
      writes: 0,
      ddr: 0,
      control: 0,
    }
  }

  /// Read from either the port or the DDR, depending on the DDR_SELECT bit in
  /// the control register.
  /// When reading the port, bor each bit, if the DDR is set to read, this
  /// reads directly from the port. If the DDR is set to write, this reads from
  /// the written value.
  pub fn read(&mut self) -> u8 {
    if self.control & pia_control_bits::DDR_SELECT != 0 {
      (self.port.read() & !self.ddr) | (self.writes & self.ddr)
    } else {
      self.ddr
    }
  }

  /// Write to either the port or the DDR, depending on the DDR_SELECT bit in
  /// the control register.
  /// Respects the DDR, so if a bit in the DDR is set to read, then that bit
  /// will not be written.
  pub fn write(&mut self, value: u8) {
    if self.control & pia_control_bits::DDR_SELECT != 0 {
      self.writes = value;
      self.port.write(value & self.ddr);
    } else {
      self.ddr = value;
    }
  }

  /// Poll the underlying port for interrupts.
  pub fn poll(&mut self, cycles_since_poll: u64, total_cycle_count: u64) -> bool {
    self.port.poll(cycles_since_poll, total_cycle_count)
  }

  /// Reset the DDR, control register, and underlying port.
  pub fn reset(&mut self) {
    self.ddr = 0;
    self.control = 0;

    self.port.reset();
  }
}

#[allow(dead_code)]
/// The meanings of each bit in the control register.
pub mod pia_control_bits {
  pub const C1_ACTIVE_TRANSITION_FLAG: u8 = 0b10000000; // 1 = 0->1, 0 = 1->0
  pub const C2_ACTIVE_TRANSITION_FLAG: u8 = 0b01000000;
  pub const C2_DIRECTION: u8 = 0b00100000; // 1 = output, 0 = input
  pub const C2_CONTROL: u8 = 0b00011000; // ???
  pub const DDR_SELECT: u8 = 0b00000100; // enable accessing DDR
  pub const C1_CONTROL: u8 = 0b00000011; // interrupt status control
}

/// The MOS 6520 Peripheral Interface Adapter (PIA), containing two ports and
/// some control lines.
pub struct Pia {
  a: PiaPortRegisters,
  b: PiaPortRegisters,
}

impl Pia {
  /// Create a new PIA with the two given port implementations.
  pub fn new(a: Box<dyn Port>, b: Box<dyn Port>) -> Self {
    Self {
      a: PiaPortRegisters::new(a),
      b: PiaPortRegisters::new(b),
    }
  }
}

impl Memory for Pia {
  fn read(&mut self, address: u16) -> u8 {
    match address % 0x04 {
      0x00 => self.a.read(),
      0x01 => self.a.control,
      0x02 => self.b.read(),
      0x03 => self.b.control,
      _ => unreachable!(),
    }
  }

  fn write(&mut self, address: u16, value: u8) {
    match address % 0x04 {
      0x00 => self.a.write(value),
      0x01 => self.a.control = value,
      0x02 => self.b.write(value),
      0x03 => self.b.control = value,
      _ => unreachable!(),
    }
  }

  fn reset(&mut self) {
    self.a.reset();
    self.b.reset();
  }

  fn poll(&mut self, cycles_since_poll: u64, total_cycle_count: u64) -> ActiveInterrupt {
    let a = self.a.poll(cycles_since_poll, total_cycle_count);
    let b = self.b.poll(cycles_since_poll, total_cycle_count);

    if a || b {
      ActiveInterrupt::IRQ
    } else {
      ActiveInterrupt::None
    }
  }
}

#[cfg(test)]
mod tests {
  use crate::memory::NullPort;

  use super::*;

  #[test]
  fn test_read() {
    let mut pia = Pia::new(Box::new(NullPort::new()), Box::new(NullPort::new()));

    // deselect the DDR
    pia.write(0x01, pia_control_bits::DDR_SELECT);

    assert_eq!(0, pia.read(0x00));
    assert_eq!(pia_control_bits::DDR_SELECT, pia.read(0x01));
    assert_eq!(0, pia.read(0x02));
    assert_eq!(0, pia.read(0x03));

    // wraps around
    assert_eq!(0, pia.read(0x04));

    // select the DDR
    pia.write(0x01, 0);

    assert_eq!(0, pia.read(0x00));
    assert_eq!(0, pia.read(0x01));
    assert_eq!(0, pia.read(0x02));
    assert_eq!(0, pia.read(0x03));
  }

  #[test]
  fn test_write() {
    let mut pia = Pia::new(Box::new(NullPort::new()), Box::new(NullPort::new()));

    // deselect the DDR
    pia.write(0x01, pia_control_bits::DDR_SELECT);

    // writes without DDR shouldn't be reflected in reads
    pia.write(0x00, 0b10101010);
    assert_eq!(0, pia.read(0x00));

    // write to the DDR
    pia.write(0x01, 0);
    pia.write(0x00, 0b11110000);

    // now, our past writes should be reflected in reads
    // (masked by the DDR)
    pia.write(0x01, pia_control_bits::DDR_SELECT);
    assert_eq!(0b10100000, pia.read(0x00));
    assert_eq!(pia_control_bits::DDR_SELECT, pia.read(0x01));

    // and future writes should be reflected in reads
    pia.write(0x00, 0b01010101);
    assert_eq!(0b01010000, pia.read(0x00));
  }
}