#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub struct CompatibilityEntry {
pub local: bool,
pub remote: bool,
pub local_state: bool,
pub remote_state: bool,
}
impl CompatibilityEntry {
#[must_use]
pub fn new(local: bool, remote: bool, local_state: bool, remote_state: bool) -> Self {
Self {
local,
remote,
local_state,
remote_state,
}
}
#[must_use]
pub fn into_u8(self) -> u8 {
let mut res = 0;
if self.local {
res |= CompatibilityTable::ENABLED_LOCAL;
}
if self.remote {
res |= CompatibilityTable::ENABLED_REMOTE;
}
if self.local_state {
res |= CompatibilityTable::LOCAL_STATE;
}
if self.remote_state {
res |= CompatibilityTable::REMOTE_STATE;
}
res
}
#[must_use]
pub fn from(value: u8) -> Self {
Self {
local: value & CompatibilityTable::ENABLED_LOCAL == CompatibilityTable::ENABLED_LOCAL,
remote: value & CompatibilityTable::ENABLED_REMOTE == CompatibilityTable::ENABLED_REMOTE,
local_state: value & CompatibilityTable::LOCAL_STATE == CompatibilityTable::LOCAL_STATE,
remote_state: value & CompatibilityTable::REMOTE_STATE == CompatibilityTable::REMOTE_STATE,
}
}
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct CompatibilityTable {
options: [u8; 256],
}
impl Default for CompatibilityTable {
fn default() -> Self {
Self { options: [0; 256] }
}
}
impl CompatibilityTable {
pub const ENABLED_LOCAL: u8 = 1;
pub const ENABLED_REMOTE: u8 = 1 << 1;
pub const LOCAL_STATE: u8 = 1 << 2;
pub const REMOTE_STATE: u8 = 1 << 3;
#[must_use]
pub fn new() -> Self {
Self::default()
}
#[must_use]
pub fn from_options(values: &[(u8, u8)]) -> Self {
let mut options = [0; 256];
for (opt, val) in values {
options[*opt as usize] = *val;
}
Self { options }
}
pub fn support_local(&mut self, option: u8) {
let mut opt = CompatibilityEntry::from(self.options[option as usize]);
opt.local = true;
self.set_option(option, opt);
}
pub fn support_remote(&mut self, option: u8) {
let mut opt = CompatibilityEntry::from(self.options[option as usize]);
opt.remote = true;
self.set_option(option, opt);
}
pub fn support(&mut self, option: u8) {
let mut opt = CompatibilityEntry::from(self.options[option as usize]);
opt.local = true;
opt.remote = true;
self.set_option(option, opt);
}
#[must_use]
pub fn get_option(&self, option: u8) -> CompatibilityEntry {
CompatibilityEntry::from(self.options[option as usize])
}
pub fn set_option(&mut self, option: u8, entry: CompatibilityEntry) {
self.options[option as usize] = entry.into_u8();
}
pub fn reset_states(&mut self) {
for opt in &mut self.options {
let mut entry = CompatibilityEntry::from(*opt);
entry.local_state = false;
entry.remote_state = false;
*opt = entry.into_u8();
}
}
}
#[cfg(test)]
mod test_compat {
use super::*;
use crate::telnet::op_option::GMCP;
#[test]
fn test_reset() {
let mut table = CompatibilityTable::default();
let entry = CompatibilityEntry::new(true, true, true, true);
assert!(entry.remote);
assert!(entry.local);
assert!(entry.remote_state);
assert!(entry.local_state);
table.set_option(GMCP, entry);
table.reset_states();
let entry = table.get_option(GMCP);
assert!(entry.remote);
assert!(entry.local);
assert!(!entry.remote_state);
assert!(!entry.local_state);
}
}