From cec37475b441be2391e42eadf0b3bd53e6801765 Mon Sep 17 00:00:00 2001 From: Niklas Saari Date: Tue, 26 Nov 2024 16:04:49 +0200 Subject: [PATCH] perf: use shifting to construct primitive integers (#379) --- src/types/integer.rs | 29 +++++++++++++++++++++-------- 1 file changed, 21 insertions(+), 8 deletions(-) diff --git a/src/types/integer.rs b/src/types/integer.rs index 6a5211cf..80787bda 100644 --- a/src/types/integer.rs +++ b/src/types/integer.rs @@ -511,11 +511,18 @@ macro_rules! integer_type_impl { return Err(crate::error::DecodeError::integer_overflow(<$t1>::BITS, codec)); } - let mut array = [0u8; BYTE_SIZE]; - let pad = if input[0] & 0x80 == 0 { 0 } else { 0xff }; - array[..BYTE_SIZE - input.len()].fill(pad); - array[BYTE_SIZE - input.len()..].copy_from_slice(input); - Ok(Self::from_be_bytes(array)) + // Use shifting to directly construct the primitive integer types + // Convert first byte with sign extension + let mut result = (input[0] as $t1) << (BYTE_SIZE - 1) * 8; + result >>= (BYTE_SIZE - input.len()) * 8; + // // Handle remaining bytes + // Add remaining bytes + for (i, &byte) in input.iter().skip(1).enumerate() { + result |= (byte as $t1) << (8 * (input.len() - 2 - i)); + } + + Ok(result) + } #[inline(always)] @@ -622,9 +629,15 @@ macro_rules! integer_type_impl { return Err(crate::error::DecodeError::integer_overflow(<$t1>::BITS, codec)); } - let mut array = [0u8; BYTE_SIZE]; - array[BYTE_SIZE - input.len()..].copy_from_slice(input); - Ok(Self::from_be_bytes(array)) + // Use shifting to directly construct the primitive integer types + let mut result: $t1 = 0; + // Calculate how many positions to shift each byte + let start_shift = (input.len() - 1) * 8; + for (i, &byte) in input.iter().enumerate() { + let shift = start_shift - (i * 8); + result |= (byte as $t1) << shift; + } + Ok(result) } // Getting signed bytes of an unsigned integer is challenging, because we don't want to truncate the value