Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add copy_*e_to_slice family #423

Closed
wants to merge 13 commits into from
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

## [Unreleased]

### Added

- Support for borsh @ 1.5 ([#416])
- `copy_le_to_slice` family to allow easier writing to pre-allocated buffers ([#423])

[#416]: https://github.com/recmo/uint/pull/416
[#423]: https://github.com/recmo/uint/pull/423

## [1.12.4] - 2024-12-16

Expand Down
2 changes: 1 addition & 1 deletion src/bits.rs
Original file line number Diff line number Diff line change
Expand Up @@ -375,7 +375,7 @@ impl<const BITS: usize, const LIMBS: usize> Uint<BITS, LIMBS> {
return Self::ZERO;
}
let rhs = rhs % BITS;
self << rhs | self >> (BITS - rhs)
(self << rhs) | (self >> (BITS - rhs))
}

/// Shifts the bits to the right by a specified amount, `rhs`, wrapping the
Expand Down
145 changes: 145 additions & 0 deletions src/bytes.rs
Original file line number Diff line number Diff line change
Expand Up @@ -345,6 +345,112 @@ impl<const BITS: usize, const LIMBS: usize> Uint<BITS, LIMBS> {
}
Some(Self::from_limbs(limbs))
}

/// Writes the little-endian representation of the [`Uint`] to the given
/// buffer. The buffer must be large enough to hold [`Self::BYTES`] bytes.
///
/// # Panics
///
/// Panics if the buffer is not large enough to hold [`Self::BYTES`] bytes.
///
/// # Returns
///
/// The number of bytes written to the buffer (always equal to
/// [`Self::BYTES`], but often useful to make explicit for encoders).
#[inline]
pub fn copy_le_bytes_to(&self, buf: &mut [u8]) -> usize {
// This is debug only. Release panics occur later in copy_from_slice
debug_assert!(
buf.len() >= Self::BYTES,
"Buffer is too small to hold the bytes of the Uint"
);

#[cfg(target_endian = "little")]
buf[..Self::BYTES].copy_from_slice(self.as_le_slice());

#[cfg(target_endian = "big")]
{
let chunks = buf[..Self::BYTES].chunks_mut(8);

self.limbs.iter().zip(chunks).for_each(|(&limb, chunk)| {
let le = limb.to_le_bytes();
chunk.copy_from_slice(&le[..chunk.len()]);
});
}

Self::BYTES
}

/// Writes the little-endian representation of the [`Uint`] to the given
/// buffer. The buffer must be large enough to hold [`Self::BYTES`] bytes.
///
/// # Returns
///
/// [`None`], if the buffer is not large enough to hold [`Self::BYTES`]
/// bytes, and does not modify the buffer.
///
/// [`Some`] with the number of bytes written to the buffer (always
/// equal to [`Self::BYTES`], but often useful to make explicit for
/// encoders).
#[inline]
pub fn checked_copy_le_bytes_to(&self, buf: &mut [u8]) -> Option<usize> {
if buf.len() < Self::BYTES {
return None;
}

Some(self.copy_le_bytes_to(buf))
}

/// Writes the big-endian representation of the [`Uint`] to the given
/// buffer. The buffer must be large enough to hold [`Self::BYTES`] bytes.
///
/// # Panics
///
/// Panics if the buffer is not large enough to hold [`Self::BYTES`] bytes.
///
/// # Returns
///
/// The number of bytes written to the buffer (always equal to
/// [`Self::BYTES`], but often useful to make explicit for encoders).
#[inline]
pub fn copy_be_bytes_to(&self, buf: &mut [u8]) -> usize {
// This is debug only. Release panics occur later in copy_from_slice
debug_assert!(
buf.len() >= Self::BYTES,
"Buffer is too small to hold the bytes of the Uint"
);

// start from the end of the slice
let chunks = buf[..Self::BYTES].rchunks_mut(8);

self.limbs.iter().zip(chunks).for_each(|(&limb, chunk)| {
let be = limb.to_be_bytes();
let copy_from = 8 - chunk.len();
chunk.copy_from_slice(&be[copy_from..]);
});

Self::BYTES
}

/// Writes the big-endian representation of the [`Uint`] to the given
/// buffer. The buffer must be large enough to hold [`Self::BYTES`] bytes.
///
/// # Returns
///
/// [`None`], if the buffer is not large enough to hold [`Self::BYTES`]
/// bytes, and does not modify the buffer.
///
/// [`Some`] with the number of bytes written to the buffer (always
/// equal to [`Self::BYTES`], but often useful to make explicit for
/// encoders).
#[inline]
pub fn checked_copy_be_bytes_to(&self, buf: &mut [u8]) -> Option<usize> {
if buf.len() < Self::BYTES {
return None;
}

Some(self.copy_be_bytes_to(buf))
}
}

/// Number of bytes required to represent the given number of bits.
Expand Down Expand Up @@ -488,4 +594,43 @@ mod tests {
});
});
}

#[test]
fn copy_to() {
const_for!(BITS in SIZES {
const LIMBS: usize = nlimbs(BITS);
const BYTES: usize = nbytes(BITS);
proptest!(|(value: Uint<BITS, LIMBS>)|{
let mut buf = [0; BYTES];
value.copy_le_bytes_to(&mut buf);
assert_eq!(buf, value.to_le_bytes::<BYTES>());
assert_eq!(value, Uint::try_from_le_slice(&buf).unwrap());

let mut buf = [0; BYTES];
value.copy_be_bytes_to(&mut buf);
assert_eq!(buf, value.to_be_bytes::<BYTES>());
assert_eq!(value, Uint::try_from_be_slice(&buf).unwrap());
})
});
}

#[test]
fn checked_copy_to() {
const_for!(BITS in SIZES {
const LIMBS: usize = nlimbs(BITS);
const BYTES: usize = nbytes(BITS);
proptest!(|(value: Uint<BITS, LIMBS>)|{
if BYTES != 0 {
let mut buf = [0; BYTES];
let too_short = buf.len() - 1;

assert_eq!(value.checked_copy_le_bytes_to(&mut buf[..too_short]), None);
assert_eq!(buf, [0; BYTES], "buffer was modified");

assert_eq!(value.checked_copy_be_bytes_to(&mut buf[..too_short]), None);
assert_eq!(buf, [0; BYTES], "buffer was modified");
}
})
});
}
}
3 changes: 3 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,9 @@
)]

#[cfg(feature = "alloc")]
#[allow(unused_imports)]
// `unused_imports` triggers on macro_use, which is required by some support
// modules.
#[macro_use]
extern crate alloc;

Expand Down
23 changes: 20 additions & 3 deletions src/support/borsh.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,9 +41,25 @@ impl<const BITS: usize, const LIMBS: usize> BorshDeserialize for Uint<BITS, LIMB
impl<const BITS: usize, const LIMBS: usize> BorshSerialize for Uint<BITS, LIMBS> {
#[inline]
fn serialize<W: io::Write>(&self, writer: &mut W) -> io::Result<()> {
// TODO: non-allocating and remove the `alloc` feature requirement
let bytes = self.as_le_bytes();
writer.write_all(&bytes[..Self::BYTES])
#[cfg(target_endian = "little")]
return writer.write_all(self.as_le_slice());

// TODO: Replace the unsafety with `generic_const_exprs` when
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do we have an issue for this filed?

Copy link
Collaborator Author

@prestwich prestwich Dec 30, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The actions bot will auto file it when the PR is merged

// available
#[cfg(target_endian = "big")]
{
let mut limbs = [0u64; LIMBS];
// SAFETY: `limbs` is known to have identical memory layout and
// alignment to `[u8; LIMBS * 8]`, which is guaranteed to safely
// contain [u8; Self::BYTES]`, as `LIMBS * 8 >= Self::BYTES`.
// Reference:
// https://doc.rust-lang.org/reference/type-layout.html#array-layout
let mut buf = unsafe {
core::slice::from_raw_parts_mut(limbs.as_mut_ptr().cast::<u8>(), Self::BYTES)
};
self.copy_le_bytes_to(&mut buf);
writer.write_all(&buf)
}
}
}

Expand All @@ -60,6 +76,7 @@ impl<const BITS: usize, const LIMBS: usize> BorshSerialize for Bits<BITS, LIMBS>
self.as_uint().serialize(writer)
}
}

#[cfg(test)]
mod test {
use super::*;
Expand Down
2 changes: 1 addition & 1 deletion src/support/postgres.rs
Original file line number Diff line number Diff line change
Expand Up @@ -221,7 +221,7 @@ impl<'a, const BITS: usize, const LIMBS: usize> FromSql<'a> for Uint<BITS, LIMBS
let mut raw = raw.to_owned();
if padding > 0 {
for i in (1..raw.len()).rev() {
raw[i] = raw[i] >> padding | raw[i - 1] << (8 - padding);
raw[i] = (raw[i] >> padding) | (raw[i - 1] << (8 - padding));
}
raw[0] >>= padding;
}
Expand Down
Loading