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
use crate::protocol::Message;
use derive_more::Display;
use thiserror::Error;

/// Represents the status of a PLC device.
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct Status {
    /// FALS *(Failure Alarm And Reset)* generated
    pub fals: bool,
    /// Fatal error generated
    pub error: bool,
    /// Operation mode
    pub mode: StatusMode,
    /// Memory status
    pub memory: StatusMemory,
}

/// An operation mode.
#[derive(Debug, Display, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum StatusMode {
    Program,
    Run,
    Monitor,
}

/// Memory status.
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct StatusMemory {
    /// Size of program memory in bytes (if available).
    pub size: Option<u16>,
    /// Whether the program memory is write protected.
    pub write_protection: bool,
}

/// An error that can occur while trying to parse `Status`.
#[derive(Debug, Clone, Error, PartialEq, Eq)]
pub enum StatusParseError {
    /// Missing mode bytes
    #[error("Missing mode bytes")]
    MissingMode,
    /// Missing memory status bytes
    #[error("Missing memory status bytes")]
    MissingMemory,
    /// Message contains an error
    #[error("Message contains an error")]
    UnparsableMessage,
    /// Mode bits could not be mapped to any known operation mode
    #[error("Unexpected mode bits: '{0}', '{1}'")]
    UnknownMode(bool, bool),
    /// Memory size bits could not be mapped to any known memory size.
    #[error("Unexpected memory size bits: '{0}', '{1}', '{2}'")]
    UnknownMemorySize(bool, bool, bool),
}

impl TryFrom<Message> for Status {
    type Error = StatusParseError;

    fn try_from(value: Message) -> Result<Self, Self::Error> {
        if value.check_device_error().is_some() {
            return Err(Self::Error::UnparsableMessage);
        }

        // skip response code
        let mut params_iter = value.params().iter().skip(2).map(|ch| *ch as u8);

        let mode_byte = params_iter
            .next()
            .zip(params_iter.next())
            .map(|(first, second)| (first & 0b1111_0000) | (second & 0b0000_1111))
            .ok_or(Self::Error::MissingMode)?;

        let fals = (mode_byte & 0b1000_0000) > 0;
        let error = (mode_byte & 0b0001_0000) > 0;
        let mode = StatusMode::parse(mode_byte)?;

        let memory = params_iter
            .next()
            .zip(params_iter.next())
            .map(|(first, second)| (first & 0b1111_0000) | (second & 0b0000_1111))
            .ok_or(Self::Error::MissingMemory)?;
        let memory = StatusMemory::parse(memory)?;

        Ok(Self {
            fals,
            error,
            mode,
            memory,
        })
    }
}

impl StatusMode {
    /// Parse the operation mode from a byte obtained using a [`StatusRead`](crate::protocol::MessageKind::StatusRead) command.
    /// # Example
    /// ```rust
    /// use hostlink::protocol::responses::status::StatusMode;
    ///
    /// // These are the bytes you'd get from a StatusRead command if the PLC is in RUN mode.
    /// let first_byte = 0b0011_0000;
    /// let second_byte = 0b0011_0010;
    ///
    /// // We need to create a single byte by using the first byte as the first 4 bits
    /// // and the second byte as the next 4 bits.
    /// let mode_byte = (first_byte & 0b1111_0000) | (second_byte & 0b0000_1111);
    ///
    /// assert_eq!(mode_byte, 0b0011_0010 /* or 50 / 0x32 */);
    /// let status = StatusMode::parse(mode_byte).unwrap();
    ///
    /// assert_eq!(status, StatusMode::Run);
    /// ```
    pub const fn parse(byte: u8) -> Result<Self, StatusParseError> {
        let first = (byte & 0b0000_0010) > 0;
        let second = (byte & 0b0000_0001) > 0;

        match (first, second) {
            (false, false) => Ok(Self::Program),
            (true, false) => Ok(Self::Run),
            (true, true) => Ok(Self::Monitor),
            _ => Err(StatusParseError::UnknownMode(first, second)),
        }
    }
}

impl StatusMemory {
    /// Parse the memory status from a byte obtained using a [`StatusRead`](crate::protocol::MessageKind::StatusRead) command.
    pub const fn parse(byte: u8) -> Result<Self, StatusParseError> {
        let first = (byte & 0b0100_0000) > 0;
        let second = (byte & 0b0010_0000) > 0;
        let third = (byte & 0b0001_0000) > 0;
        let write_protection = (byte & 0b0000_1000) == 0;

        let program_area = match (first, second, third) {
            (false, false, false) => None,
            (false, false, true) => Some(4000),
            (false, true, false) => Some(8000),
            (true, false, false) => Some(7200),
            _ => return Err(StatusParseError::UnknownMemorySize(first, second, third)),
        };

        Ok(Self {
            size: program_area,
            write_protection,
        })
    }
}