From 080b605bbcc837baf0351da6c25b265cca13c736 Mon Sep 17 00:00:00 2001 From: "Cliff L. Biffle" Date: Mon, 2 Dec 2024 15:07:07 -0800 Subject: [PATCH] oxide-barcode: allow shorter serial and part numbers This retains the original maximum length for both fields, but permits parsed strings to contain shorter sections. Shorter sections are right-padded with NULs for consistency with IPCC. Fixes #1893 --- lib/host-sp-messages/src/lib.rs | 4 ++ lib/oxide-barcode/src/lib.rs | 99 ++++++++++++++++++++++++--------- 2 files changed, 77 insertions(+), 26 deletions(-) diff --git a/lib/host-sp-messages/src/lib.rs b/lib/host-sp-messages/src/lib.rs index 41b0f90a5e..5981290bb1 100644 --- a/lib/host-sp-messages/src/lib.rs +++ b/lib/host-sp-messages/src/lib.rs @@ -484,6 +484,10 @@ impl From for Identity { ); let mut new_id = Self::default(); + // The incoming part number and serial are already nul-padded if they're + // shorter than the allocated space in VpdIdentity, so we can merely + // copy them into the start of our fields and the result is still + // nul-padded. new_id.model[..id.part_number.len()].copy_from_slice(&id.part_number); new_id.revision = id.revision; new_id.serial[..id.serial.len()].copy_from_slice(&id.serial); diff --git a/lib/oxide-barcode/src/lib.rs b/lib/oxide-barcode/src/lib.rs index 3808860d88..9163b75e11 100644 --- a/lib/oxide-barcode/src/lib.rs +++ b/lib/oxide-barcode/src/lib.rs @@ -60,6 +60,8 @@ impl VpdIdentity { return Err(ParseError::UnexpectedFields); } + // Note: the fact that this is created _zeroed_ is important for the + // variable length field handling below. let mut out = VpdIdentity::new_zeroed(); match version { @@ -75,10 +77,12 @@ impl VpdIdentity { } // V2 part number includes the hyphen; copy it as-is. b"OXV2" | b"0XV2" => { - if part_number.len() != out.part_number.len() { + if part_number.len() > out.part_number.len() { return Err(ParseError::WrongPartNumberLength); } - out.part_number.copy_from_slice(part_number); + out.part_number[..part_number.len()] + .copy_from_slice(part_number); + // tail is already zeroed } _ => return Err(ParseError::UnknownVersion), } @@ -88,11 +92,11 @@ impl VpdIdentity { .and_then(|rev| rev.parse().ok()) .ok_or(ParseError::BadRevision)?; - if serial.len() != out.serial.len() { + if serial.len() > out.serial.len() { return Err(ParseError::WrongSerialLength); } - - out.serial.copy_from_slice(serial); + out.serial[..serial.len()].copy_from_slice(serial); + // tail is already zeroed Ok(out) } @@ -102,39 +106,82 @@ impl VpdIdentity { mod tests { use super::*; - #[test] - fn parse_oxv1() { - let expected = VpdIdentity { - part_number: *b"123-0000456", - revision: 23, - serial: *b"TST01234567", - }; - + #[track_caller] + fn check_parse(input: &[u8], expected: VpdIdentity) { assert_eq!( expected, - VpdIdentity::parse(b"0XV1:1230000456:023:TST01234567").unwrap() + VpdIdentity::parse(input).unwrap(), + "parsing string: {}", + String::from_utf8_lossy(input), ); + + // We accept barcode strings that start with both leading zero and + // leading capital-O. Permute our input from one of these to the other + // to make sure both forms parse equivalently. + let mut copy = input.to_vec(); + match copy[0] { + b'0' => copy[0] = b'O', + b'O' => copy[0] = b'0', + c => { + panic!("unexpected leading character in string: {}", c as char) + } + } + assert_eq!( expected, - VpdIdentity::parse(b"OXV1:1230000456:023:TST01234567").unwrap() + VpdIdentity::parse(©).unwrap(), + "parsing string: {}", + String::from_utf8_lossy(©), + ); + } + + #[test] + fn parse_oxv1() { + check_parse( + b"0XV1:1230000456:023:TST01234567", + VpdIdentity { + part_number: *b"123-0000456", + revision: 23, + serial: *b"TST01234567", + }, ); } #[test] fn parse_oxv2() { - let expected = VpdIdentity { - part_number: *b"123-0000456", - revision: 23, - serial: *b"TST01234567", - }; + check_parse( + b"0XV2:123-0000456:023:TST01234567", + VpdIdentity { + part_number: *b"123-0000456", + revision: 23, + serial: *b"TST01234567", + }, + ); + } - assert_eq!( - expected, - VpdIdentity::parse(b"0XV2:123-0000456:023:TST01234567").unwrap() + #[test] + fn parse_oxv2_shorter_serial() { + check_parse( + b"0XV2:123-0000456:023:TST0123456", + VpdIdentity { + part_number: *b"123-0000456", + revision: 23, + // should get padded with NULs to the right: + serial: *b"TST0123456\0", + }, ); - assert_eq!( - expected, - VpdIdentity::parse(b"OXV2:123-0000456:023:TST01234567").unwrap() + } + + #[test] + fn parse_oxv2_shorter_part() { + check_parse( + b"0XV2:123-000045:023:TST01234567", + VpdIdentity { + // should get padded with NULs to the right: + part_number: *b"123-000045\0", + revision: 23, + serial: *b"TST01234567", + }, ); } }