From aa956bfc7d4776321ccf4cb89a22757d2e477635 Mon Sep 17 00:00:00 2001 From: Vivian Date: Wed, 23 Oct 2024 22:45:09 +0200 Subject: [PATCH] fix soundness bug --- src/with_const_generics.rs | 153 ++++++++++++++++++------------------- 1 file changed, 75 insertions(+), 78 deletions(-) diff --git a/src/with_const_generics.rs b/src/with_const_generics.rs index 4f76e23..bc70007 100644 --- a/src/with_const_generics.rs +++ b/src/with_const_generics.rs @@ -1,8 +1,8 @@ use crate::ringbuffer_trait::{RingBufferIntoIterator, RingBufferIterator, RingBufferMutIterator}; use crate::RingBuffer; use core::iter::FromIterator; -use core::mem; use core::mem::MaybeUninit; +use core::mem::{self, ManuallyDrop}; use core::ops::{Index, IndexMut}; /// The `ConstGenericRingBuffer` struct is a `RingBuffer` implementation which does not require `alloc` but @@ -42,11 +42,14 @@ pub struct ConstGenericRingBuffer { impl From<[T; CAP]> for ConstGenericRingBuffer { fn from(value: [T; CAP]) -> Self { + let v = ManuallyDrop::new(value); Self { // Safety: // T has the same layout as MaybeUninit // [T; N] has the same layout as [MaybeUninit; N] - buf: unsafe { mem::transmute_copy(&value) }, + // Without ManuallyDrop this would be unsound as + // transmute_copy doesn't take ownership + buf: unsafe { mem::transmute_copy(&v) }, readptr: 0, writeptr: CAP, } @@ -183,7 +186,7 @@ impl ConstGenericRingBuffer { // https://rust-lang.github.io/rust-clippy/master/index.html#uninit_assumed_init #[allow(clippy::uninit_assumed_init)] Self { - buf: unsafe { MaybeUninit::uninit().assume_init() }, + buf: [const { MaybeUninit::::uninit() }; CAP], writeptr: 0, readptr: 0, } @@ -280,7 +283,6 @@ unsafe impl RingBuffer for ConstGenericRingBuffer IndexMut for ConstGenericRingBuffer { #[cfg(test)] mod tests { - use super::*; + use crate::{AllocRingBuffer, ConstGenericRingBuffer, GrowableAllocRingBuffer, RingBuffer}; + use alloc::collections::{LinkedList, VecDeque}; + use alloc::string::ToString; + use alloc::vec; #[test] fn test_not_power_of_two() { @@ -428,79 +433,71 @@ mod tests { } } - #[cfg(test)] - mod tests { - use crate::{AllocRingBuffer, ConstGenericRingBuffer, GrowableAllocRingBuffer, RingBuffer}; - use alloc::collections::{LinkedList, VecDeque}; - use alloc::string::ToString; - use alloc::vec; - - #[test] - fn from() { - assert_eq!( - ConstGenericRingBuffer::::from([1, 2, 3]).to_vec(), - vec![1, 2, 3] - ); - - let v: &[i32; 3] = &[1, 2, 3]; - assert_eq!( - ConstGenericRingBuffer::::from(v).to_vec(), - vec![1, 2, 3] - ); - - let v: &[i32] = &[1, 2, 3]; - assert_eq!( - ConstGenericRingBuffer::::from(v).to_vec(), - vec![1, 2, 3] - ); - - let v: &mut [i32; 3] = &mut [1, 2, 3]; - assert_eq!( - ConstGenericRingBuffer::::from(v).to_vec(), - vec![1, 2, 3] - ); - - let v: &mut [i32] = &mut [1, 2, 3]; - assert_eq!( - ConstGenericRingBuffer::::from(v).to_vec(), - vec![1, 2, 3] - ); - - assert_eq!( - ConstGenericRingBuffer::::from(vec![1, 2, 3]).to_vec(), - vec![1, 2, 3] - ); - assert_eq!( - ConstGenericRingBuffer::::from( - vec![1, 2, 3].into_iter().collect::>() - ) - .to_vec(), - vec![1, 2, 3] - ); - assert_eq!( - ConstGenericRingBuffer::::from( - vec![1, 2, 3].into_iter().collect::>() - ) + #[test] + fn from() { + assert_eq!( + ConstGenericRingBuffer::::from([1, 2, 3]).to_vec(), + vec![1, 2, 3] + ); + + let v: &[i32; 3] = &[1, 2, 3]; + assert_eq!( + ConstGenericRingBuffer::::from(v).to_vec(), + vec![1, 2, 3] + ); + + let v: &[i32] = &[1, 2, 3]; + assert_eq!( + ConstGenericRingBuffer::::from(v).to_vec(), + vec![1, 2, 3] + ); + + let v: &mut [i32; 3] = &mut [1, 2, 3]; + assert_eq!( + ConstGenericRingBuffer::::from(v).to_vec(), + vec![1, 2, 3] + ); + + let v: &mut [i32] = &mut [1, 2, 3]; + assert_eq!( + ConstGenericRingBuffer::::from(v).to_vec(), + vec![1, 2, 3] + ); + + assert_eq!( + ConstGenericRingBuffer::::from(vec![1, 2, 3]).to_vec(), + vec![1, 2, 3] + ); + assert_eq!( + ConstGenericRingBuffer::::from( + vec![1, 2, 3].into_iter().collect::>() + ) + .to_vec(), + vec![1, 2, 3] + ); + assert_eq!( + ConstGenericRingBuffer::::from( + vec![1, 2, 3].into_iter().collect::>() + ) + .to_vec(), + vec![1, 2, 3] + ); + assert_eq!( + ConstGenericRingBuffer::<_, 3>::from("abc".to_string()).to_vec(), + vec!['a', 'b', 'c'] + ); + assert_eq!( + ConstGenericRingBuffer::<_, 3>::from("abc").to_vec(), + vec!['a', 'b', 'c'] + ); + assert_eq!( + ConstGenericRingBuffer::<_, 3>::from(GrowableAllocRingBuffer::from(vec![1, 2, 3])) .to_vec(), - vec![1, 2, 3] - ); - assert_eq!( - ConstGenericRingBuffer::<_, 3>::from("abc".to_string()).to_vec(), - vec!['a', 'b', 'c'] - ); - assert_eq!( - ConstGenericRingBuffer::<_, 3>::from("abc").to_vec(), - vec!['a', 'b', 'c'] - ); - assert_eq!( - ConstGenericRingBuffer::<_, 3>::from(GrowableAllocRingBuffer::from(vec![1, 2, 3])) - .to_vec(), - vec![1, 2, 3] - ); - assert_eq!( - ConstGenericRingBuffer::<_, 3>::from(AllocRingBuffer::from(vec![1, 2, 3])).to_vec(), - vec![1, 2, 3] - ); - } + vec![1, 2, 3] + ); + assert_eq!( + ConstGenericRingBuffer::<_, 3>::from(AllocRingBuffer::from(vec![1, 2, 3])).to_vec(), + vec![1, 2, 3] + ); } }