summaryrefslogtreecommitdiff
path: root/src/vrrpv2.rs
diff options
context:
space:
mode:
Diffstat (limited to 'src/vrrpv2.rs')
-rw-r--r--src/vrrpv2.rs84
1 files changed, 39 insertions, 45 deletions
diff --git a/src/vrrpv2.rs b/src/vrrpv2.rs
index 8c2d7d6..b6aa82f 100644
--- a/src/vrrpv2.rs
+++ b/src/vrrpv2.rs
@@ -115,58 +115,28 @@ fn parse(input: &[u8]) -> Result<VRRPv2, VRRPv2Error> {
})
}
-// Nightly has a nicer array_chunks API to express it more succinctly.
-// let mut chunks = bytes.array_chunks(2);
-// let mut sum = chunks.map(u16::from_ne_bytes).map(|b| b as u32).sum::<u32>();
-// // handle the remainder
-// if let Some([b]) = chunks.remainder() {
-// sum += *b as u32
-// }
-
-// Shadowing can be used to avoid `mut`...
-// let sum =...;
-// let sum = (sum & 0xffff) + (sum >> 16);
-// let sum = (sum & 0xffff) + (sum >> 16);
-// manually un-rolling while loop since it's needed atmost twice for an u32.
-fn validate_checksum(bytes: &[u8]) -> bool {
- let mut sum: u32 = bytes.chunks(2).fold(0, |acc: u32, x| {
- acc + u32::from(u16::from_ne_bytes(x.try_into().unwrap()))
- });
+// nightly has as_chunks that allows for a nicer code...
+// let (chunks, remainder) = bytes.as_chunks(2);
+// fold chunks and remainder without an if.
+fn checksum(bytes: &[u8]) -> u16 {
+ let mut sum: u32 = 0;
+ for chunk in bytes.chunks(2) {
+ // Left over byte if any
+ if chunk.len() == 1 {
+ sum += u32::from(chunk[0]);
+ } else {
+ sum += u32::from(u16::from_be_bytes(chunk.try_into().unwrap()));
+ }
+ }
while (sum >> 16) > 0 {
sum = (sum & 0xffff) + (sum >> 16);
}
- let checksum = !(sum as u16);
- checksum == 0
+ !(sum as u16)
}
-/// Parse and validate a byte array to construct a VRRPv2 struct.
-///
-/// # Examples
-///
-/// ```
-/// use vrrpd::vrrpv2::VRRPv2;
-/// use vrrpd::vrrpv2::VRRPv2AuthType;
-/// use vrrpd::vrrpv2::from_bytes;
-/// use std::net::Ipv4Addr;
-///
-/// let bytes = [
-/// 0x21, 0x01, 0x64, 0x01, 0x00, 0x01, 0xba, 0x52, 0xc0, 0xa8, 0x00, 0x01, 0x00, 0x00, 0x00,
-/// 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
-/// ];
-/// let expected = VRRPv2 {
-/// virtual_router_id: 1,
-/// priority: 100,
-/// count_ip_addrs: 1,
-/// auth_type: VRRPv2AuthType::VRRPv2AuthNoAuth,
-/// checksum: 47698,
-/// advertisement_interval: 1,
-/// ip_addrs: vec![Ipv4Addr::from([192, 168, 0, 1])],
-/// };
-/// assert_eq!(from_bytes(&bytes), Ok(expected));
-/// ```
pub fn from_bytes(bytes: &[u8]) -> Result<VRRPv2, VRRPv2Error> {
let vrrpv2 = parse(bytes)?;
- if !validate_checksum(bytes) {
+ if checksum(bytes) != 0 {
return Err(VRRPv2Error::InvalidChecksum);
}
Ok(vrrpv2)
@@ -217,3 +187,27 @@ fn test_invalid_checksum() {
];
assert_eq!(from_bytes(&bytes), Err(VRRPv2Error::InvalidChecksum));
}
+
+#[test]
+fn test_checksum() {
+ let bytes = [0x00, 0x01, 0xf2, 0x03, 0xf4, 0xf5, 0xf6, 0xf7];
+ assert_eq!(checksum(&bytes), 0x220d);
+}
+
+#[test]
+fn test_checksum_singlebyte() {
+ let bytes = [0; 1];
+ assert_eq!(checksum(&bytes), 0xffff);
+}
+
+#[test]
+fn test_checksum_twobytes() {
+ let bytes = [0x00, 0xff];
+ assert_eq!(checksum(&bytes), 0xff00);
+}
+
+#[test]
+fn test_checksum_another() {
+ let bytes = [0xe3, 0x4f, 0x23, 0x96, 0x44, 0x27, 0x99, 0xf3];
+ assert_eq!(checksum(&bytes), 0x1aff);
+}