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
use std::collections::HashMap;
use std::fmt;
use std::fmt::{Debug, Display};

/// The registers of the machine.
pub trait MachineRegister: Debug + Display {
    /// Reads a register.
    fn read(&self, key: char) -> i64;

    /// Writes a register.
    fn write(&mut self, key: char, value: i64);

    /// Clears the register.
    fn clear(&mut self);

    /// Increments (or decrements if by < 0) a register.
    fn increment(&mut self, key: char, by: i64) {
        self.write(key, self.read(key) + by)
    }

    /// Writes a boolean value (1 = true, 0 = false) to a register.
    fn write_bool(&mut self, key: char, condition: bool) {
        self.write(key, if condition { 1 } else { 0 });
    }
}

/// The default implementation using a HashMap.
#[derive(Debug)]
pub struct HashMapRegister {
    registers: HashMap<char, i64>,
}

impl HashMapRegister {
    /// Constructs a new HashMapRegister.
    pub fn new() -> Self {
        HashMapRegister {
            registers: HashMap::new(),
        }
    }
}

impl Default for HashMapRegister {
    fn default() -> Self {
        HashMapRegister::new()
    }
}

impl fmt::Display for HashMapRegister {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        let mut keys: Vec<char> = self.registers.keys().copied().collect();
        keys.sort_unstable();

        write!(f, "[")?;
        for (i, key) in keys.iter().enumerate() {
            if i != 0 {
                write!(f, ", ")?;
            }
            write!(f, "{}=", key)?;
            fmt::Display::fmt(&self.read(*key), f)?;
        }
        write!(f, "]")
    }
}

impl MachineRegister for HashMapRegister {
    fn read(&self, key: char) -> i64 {
        *self.registers.get(&key).unwrap_or(&0)
    }

    fn write(&mut self, key: char, value: i64) {
        *self.registers.entry(key).or_insert(0) = value;
    }

    fn clear(&mut self) {
        self.registers.clear();
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_read_empty_register() {
        let registers = HashMapRegister::new();
        assert_eq!(registers.read('a'), 0);
    }

    #[test]
    fn test_write_and_read_register() {
        let mut registers = HashMapRegister::new();
        registers.write('a', 10);
        assert_eq!(registers.read('a'), 10);
    }

    #[test]
    fn test_increment_empty_register() {
        let mut registers = HashMapRegister::new();
        registers.increment('a', 5);
        assert_eq!(registers.read('a'), 5);
    }

    #[test]
    fn test_increment_non_empty_register() {
        let mut registers = HashMapRegister::new();
        registers.write('a', 5);
        registers.increment('a', 5);
        assert_eq!(registers.read('a'), 10);
    }

    #[test]
    fn test_format_register() {
        let mut registers = HashMapRegister::new();
        registers.write('a', 10);
        registers.write('d', -5);
        registers.write('c', 12);
        registers.write('b', 0);
        assert_eq!(format!("{}", registers), "[a=10, b=0, c=12, d=-5]");
    }
}