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
use std::{cell::Cell, rc::Rc};

use super::{ActiveInterrupt, Memory};

/// Represents the memory banking features found in the Commodore 64 and other
/// devices. Multiple memory implementations are all mapped to the same
/// address space. The active implementation is selected by external logic.
pub struct BankedMemory {
  banks: Vec<Box<dyn Memory>>,
  active: Rc<Cell<usize>>,
}

impl BankedMemory {
  /// Create a new, empty banked memory.
  pub fn new(active: Rc<Cell<usize>>) -> Self {
    Self {
      banks: Vec::new(),
      active,
    }
  }

  /// Add a new memory implementation to the banked memory.
  pub fn bank(mut self, memory: impl Memory + 'static) -> Self {
    self.banks.push(Box::new(memory));

    self
  }
}

impl Memory for BankedMemory {
  fn read(&mut self, address: u16) -> u8 {
    match self.banks.get_mut(self.active.get()) {
      Some(memory) => memory.read(address),
      None => panic!("Invalid bank {} selected", self.active.get()),
    }
  }

  fn write(&mut self, address: u16, value: u8) {
    match self.banks.get_mut(self.active.get()) {
      Some(memory) => memory.write(address, value),
      None => panic!("Invalid bank {} selected", self.active.get()),
    }
  }

  fn reset(&mut self) {
    for memory in self.banks.iter_mut() {
      memory.reset();
    }
  }

  fn poll(&mut self, cycles_since_poll: u64, total_cycle_count: u64) -> ActiveInterrupt {
    let mut highest = ActiveInterrupt::None;

    for mapped in &mut self.banks {
      let interrupt = mapped.poll(cycles_since_poll, total_cycle_count);

      match interrupt {
        ActiveInterrupt::None => (),
        ActiveInterrupt::NMI => {
          highest = ActiveInterrupt::NMI;
        }
        ActiveInterrupt::IRQ => {
          highest = ActiveInterrupt::IRQ;
        }
      }
    }

    highest
  }
}