Introduce Storage::forget_elements() to fix memory leak in Matrix::generic_resize() (#1416)
* Add Storage::forget * Adjust implementations of Reallocator to use Storage::forget * Fix formatting * Rename forget to forget_elements and add safety comments * Update comments in Reallocator implementations --------- Co-authored-by: Nick Mertin <nickmertin@gmail.com>
This commit is contained in:
parent
eb228faa2b
commit
5ad68f486d
|
@ -130,6 +130,12 @@ where
|
|||
{
|
||||
self.clone()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn forget_elements(self) {
|
||||
// No additional cleanup required.
|
||||
std::mem::forget(self);
|
||||
}
|
||||
}
|
||||
|
||||
unsafe impl<T, const R: usize, const C: usize> RawStorageMut<T, Const<R>, Const<C>>
|
||||
|
|
|
@ -15,7 +15,7 @@ use crate::base::array_storage::ArrayStorage;
|
|||
use crate::base::dimension::Dim;
|
||||
#[cfg(any(feature = "alloc", feature = "std"))]
|
||||
use crate::base::dimension::{DimName, Dyn};
|
||||
use crate::base::storage::{RawStorage, RawStorageMut};
|
||||
use crate::base::storage::{RawStorage, RawStorageMut, Storage};
|
||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||
use crate::base::vec_storage::VecStorage;
|
||||
use crate::base::Scalar;
|
||||
|
@ -210,8 +210,8 @@ where
|
|||
|
||||
// Safety:
|
||||
// - We don’t care about dropping elements because the caller is responsible for dropping things.
|
||||
// - We forget `buf` so that we don’t drop the other elements.
|
||||
std::mem::forget(buf);
|
||||
// - We forget `buf` so that we don’t drop the other elements, but ensure the buffer itself is cleaned up.
|
||||
buf.forget_elements();
|
||||
|
||||
res
|
||||
}
|
||||
|
@ -241,8 +241,8 @@ where
|
|||
|
||||
// Safety:
|
||||
// - We don’t care about dropping elements because the caller is responsible for dropping things.
|
||||
// - We forget `buf` so that we don’t drop the other elements.
|
||||
std::mem::forget(buf);
|
||||
// - We forget `buf` so that we don’t drop the other elements, but ensure the buffer itself is cleaned up.
|
||||
buf.forget_elements();
|
||||
|
||||
res
|
||||
}
|
||||
|
@ -272,8 +272,8 @@ where
|
|||
|
||||
// Safety:
|
||||
// - We don’t care about dropping elements because the caller is responsible for dropping things.
|
||||
// - We forget `buf` so that we don’t drop the other elements.
|
||||
std::mem::forget(buf);
|
||||
// - We forget `buf` so that we don’t drop the other elements, but ensure the buffer itself is cleaned up.
|
||||
buf.forget_elements();
|
||||
|
||||
res
|
||||
}
|
||||
|
|
|
@ -229,6 +229,11 @@ macro_rules! storage_impl(
|
|||
let it = MatrixIter::new(self).cloned();
|
||||
DefaultAllocator::allocate_from_iterator(nrows, ncols, it)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn forget_elements(self) {
|
||||
// No cleanup required.
|
||||
}
|
||||
}
|
||||
)*}
|
||||
);
|
||||
|
|
|
@ -149,6 +149,9 @@ pub unsafe trait Storage<T: Scalar, R: Dim, C: Dim = U1>: RawStorage<T, R, C> {
|
|||
fn clone_owned(&self) -> Owned<T, R, C>
|
||||
where
|
||||
DefaultAllocator: Allocator<R, C>;
|
||||
|
||||
/// Drops the storage without calling the destructors on the contained elements.
|
||||
fn forget_elements(self);
|
||||
}
|
||||
|
||||
/// Trait implemented by matrix data storage that can provide a mutable access to its elements.
|
||||
|
|
|
@ -281,6 +281,17 @@ where
|
|||
{
|
||||
self.clone()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn forget_elements(mut self) {
|
||||
// SAFETY: setting the length to zero is always sound, as it does not
|
||||
// cause any memory to be deemed initialized. If the previous length was
|
||||
// non-zero, it is equivalent to using mem::forget to leak each element.
|
||||
// Then, when this function returns, self.data is dropped, freeing the
|
||||
// allocated memory, but the elements are not dropped because they are
|
||||
// now considered uninitialized.
|
||||
unsafe { self.data.set_len(0) };
|
||||
}
|
||||
}
|
||||
|
||||
unsafe impl<T, R: DimName> RawStorage<T, R, Dyn> for VecStorage<T, R, Dyn> {
|
||||
|
@ -332,6 +343,17 @@ where
|
|||
{
|
||||
self.clone()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn forget_elements(mut self) {
|
||||
// SAFETY: setting the length to zero is always sound, as it does not
|
||||
// cause any memory to be deemed initialized. If the previous length was
|
||||
// non-zero, it is equivalent to using mem::forget to leak each element.
|
||||
// Then, when this function returns, self.data is dropped, freeing the
|
||||
// allocated memory, but the elements are not dropped because they are
|
||||
// now considered uninitialized.
|
||||
unsafe { self.data.set_len(0) };
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
|
|
Loading…
Reference in New Issue