2017-02-13 01:17:09 +08:00
|
|
|
//! Abstract definition of a matrix data storage.
|
|
|
|
|
2021-06-18 08:51:00 +08:00
|
|
|
use std::ptr;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2021-08-03 00:41:46 +08:00
|
|
|
use crate::base::allocator::{Allocator, SameShapeC, SameShapeR};
|
2019-03-23 21:29:07 +08:00
|
|
|
use crate::base::default_allocator::DefaultAllocator;
|
|
|
|
use crate::base::dimension::{Dim, U1};
|
2021-08-03 00:41:46 +08:00
|
|
|
use crate::base::Scalar;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
/*
|
2017-08-03 01:37:44 +08:00
|
|
|
* Aliases for allocation results.
|
2016-12-05 05:44:42 +08:00
|
|
|
*/
|
2021-08-03 00:41:46 +08:00
|
|
|
/// The data storage for the sum of two matrices with dimensions `(R1, C1)` and `(R2, C2)`.
|
|
|
|
pub type SameShapeStorage<T, R1, C1, R2, C2> =
|
|
|
|
<DefaultAllocator as Allocator<T, SameShapeR<R1, R2>, SameShapeC<C1, C2>>>::Buffer;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2020-11-15 23:57:49 +08:00
|
|
|
// TODO: better name than Owned ?
|
2017-08-03 01:37:44 +08:00
|
|
|
/// The owned data storage that can be allocated from `S`.
|
2021-08-03 00:41:46 +08:00
|
|
|
pub type Owned<T, R, C = U1> = <DefaultAllocator as Allocator<T, R, C>>::Buffer;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2021-08-03 00:41:46 +08:00
|
|
|
/// The owned data storage that can be allocated from `S`.
|
|
|
|
pub type OwnedUninit<T, R, C = U1> = <DefaultAllocator as Allocator<T, R, C>>::BufferUninit;
|
2021-07-30 01:33:45 +08:00
|
|
|
|
2017-08-03 01:37:44 +08:00
|
|
|
/// The row-stride of the owned data storage for a buffer of dimension `(R, C)`.
|
2021-08-03 00:41:46 +08:00
|
|
|
pub type RStride<T, R, C = U1> =
|
|
|
|
<<DefaultAllocator as Allocator<T, R, C>>::Buffer as RawStorage<T, R, C>>::RStride;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2017-08-03 01:37:44 +08:00
|
|
|
/// The column-stride of the owned data storage for a buffer of dimension `(R, C)`.
|
2021-08-03 00:41:46 +08:00
|
|
|
pub type CStride<T, R, C = U1> =
|
|
|
|
<<DefaultAllocator as Allocator<T, R, C>>::Buffer as RawStorage<T, R, C>>::CStride;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
/// The trait shared by all matrix data storage.
|
|
|
|
///
|
2020-11-15 23:57:49 +08:00
|
|
|
/// TODO: doc
|
2016-12-05 05:44:42 +08:00
|
|
|
///
|
|
|
|
/// Note that `Self` must always have a number of elements compatible with the matrix length (given
|
|
|
|
/// by `R` and `C` if they are known at compile-time). For example, implementors of this trait
|
|
|
|
/// should **not** allow the user to modify the size of the underlying buffer with safe methods
|
2018-12-06 05:46:17 +08:00
|
|
|
/// (for example the `VecStorage::data_mut` method is unsafe because the user could change the
|
2016-12-05 05:44:42 +08:00
|
|
|
/// vector's size so that it no longer contains enough elements: this will lead to UB.
|
2021-08-03 00:41:46 +08:00
|
|
|
pub unsafe trait RawStorage<T, R: Dim, C: Dim = U1>: Sized {
|
2016-12-05 05:44:42 +08:00
|
|
|
/// The static stride of this storage's rows.
|
|
|
|
type RStride: Dim;
|
|
|
|
|
|
|
|
/// The static stride of this storage's columns.
|
|
|
|
type CStride: Dim;
|
|
|
|
|
|
|
|
/// The matrix data pointer.
|
2021-04-11 17:00:38 +08:00
|
|
|
fn ptr(&self) -> *const T;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
/// The dimension of the matrix at run-time. Arr length of zero indicates the additive identity
|
|
|
|
/// element of any dimension. Must be equal to `Self::dimension()` if it is not `None`.
|
|
|
|
fn shape(&self) -> (R, C);
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// The spacing between consecutive row elements and consecutive column elements.
|
2016-12-05 05:44:42 +08:00
|
|
|
///
|
|
|
|
/// For example this returns `(1, 5)` for a row-major matrix with 5 columns.
|
|
|
|
fn strides(&self) -> (Self::RStride, Self::CStride);
|
|
|
|
|
|
|
|
/// Compute the index corresponding to the irow-th row and icol-th column of this matrix. The
|
|
|
|
/// index must be such that the following holds:
|
|
|
|
///
|
2021-07-07 10:05:25 +08:00
|
|
|
/// ```ignore
|
2016-12-05 05:44:42 +08:00
|
|
|
/// let lindex = self.linear_index(irow, icol);
|
2020-11-16 00:25:29 +08:00
|
|
|
/// assert!(*self.get_unchecked(irow, icol) == *self.get_unchecked_linear(lindex))
|
2016-12-05 05:44:42 +08:00
|
|
|
/// ```
|
|
|
|
#[inline]
|
|
|
|
fn linear_index(&self, irow: usize, icol: usize) -> usize {
|
|
|
|
let (rstride, cstride) = self.strides();
|
|
|
|
|
|
|
|
irow * rstride.value() + icol * cstride.value()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Gets the address of the i-th matrix component without performing bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, dereferencing the result will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-07-09 00:12:43 +08:00
|
|
|
fn get_address_unchecked_linear(&self, i: usize) -> *const T {
|
2020-11-19 18:56:58 +08:00
|
|
|
self.ptr().wrapping_add(i)
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Gets the address of the i-th matrix component without performing bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, dereferencing the result will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-07-09 00:12:43 +08:00
|
|
|
fn get_address_unchecked(&self, irow: usize, icol: usize) -> *const T {
|
2016-12-05 05:44:42 +08:00
|
|
|
self.get_address_unchecked_linear(self.linear_index(irow, icol))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves a reference to the i-th element without bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, the method will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
unsafe fn get_unchecked_linear(&self, i: usize) -> &T {
|
2016-12-05 05:44:42 +08:00
|
|
|
&*self.get_address_unchecked_linear(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves a reference to the i-th element without bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, the method will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
unsafe fn get_unchecked(&self, irow: usize, icol: usize) -> &T {
|
2016-12-05 05:44:42 +08:00
|
|
|
self.get_unchecked_linear(self.linear_index(irow, icol))
|
|
|
|
}
|
2017-08-03 01:37:44 +08:00
|
|
|
|
|
|
|
/// Indicates whether this data buffer stores its elements contiguously.
|
2021-06-17 15:46:49 +08:00
|
|
|
///
|
2021-07-04 11:19:07 +08:00
|
|
|
/// # Safety
|
2021-07-09 00:12:43 +08:00
|
|
|
/// This function must not return `true` if the underlying storage is not contiguous,
|
|
|
|
/// or undefined behaviour will occur.
|
|
|
|
fn is_contiguous(&self) -> bool;
|
2017-08-03 01:37:44 +08:00
|
|
|
|
|
|
|
/// Retrieves the data buffer as a contiguous slice.
|
|
|
|
///
|
2021-07-06 05:51:27 +08:00
|
|
|
/// # Safety
|
2017-08-03 01:37:44 +08:00
|
|
|
/// The matrix components may not be stored in a contiguous way, depending on the strides.
|
2021-06-17 15:46:49 +08:00
|
|
|
/// This method is unsafe because this can yield to invalid aliasing when called on some pairs
|
|
|
|
/// of matrix slices originating from the same matrix with strides.
|
|
|
|
///
|
|
|
|
/// Call the safe alternative `matrix.as_slice()` instead.
|
|
|
|
unsafe fn as_slice_unchecked(&self) -> &[T];
|
2021-08-03 00:41:46 +08:00
|
|
|
}
|
2017-08-03 01:37:44 +08:00
|
|
|
|
2021-08-03 00:41:46 +08:00
|
|
|
pub unsafe trait Storage<T, R: Dim, C: Dim = U1>: RawStorage<T, R, C> {
|
2017-08-03 01:37:44 +08:00
|
|
|
/// Builds a matrix data storage that does not contain any reference.
|
2021-04-11 17:00:38 +08:00
|
|
|
fn into_owned(self) -> Owned<T, R, C>
|
2020-04-06 00:49:48 +08:00
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
DefaultAllocator: Allocator<T, R, C>;
|
2017-08-03 01:37:44 +08:00
|
|
|
|
2017-08-14 01:53:02 +08:00
|
|
|
/// Clones this data storage to one that does not contain any reference.
|
2021-04-11 17:00:38 +08:00
|
|
|
fn clone_owned(&self) -> Owned<T, R, C>
|
2020-04-06 00:49:48 +08:00
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
DefaultAllocator: Allocator<T, R, C>;
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2017-02-13 01:17:09 +08:00
|
|
|
/// Trait implemented by matrix data storage that can provide a mutable access to its elements.
|
|
|
|
///
|
|
|
|
/// Note that a mutable access does not mean that the matrix owns its data. For example, a mutable
|
|
|
|
/// matrix slice can provide mutable access to its elements even if it does not own its data (it
|
|
|
|
/// contains only an internal reference to them).
|
2021-08-03 00:41:46 +08:00
|
|
|
pub unsafe trait RawStorageMut<T, R: Dim, C: Dim = U1>: RawStorage<T, R, C> {
|
2016-12-05 05:44:42 +08:00
|
|
|
/// The matrix mutable data pointer.
|
2021-04-11 17:00:38 +08:00
|
|
|
fn ptr_mut(&mut self) -> *mut T;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
/// Gets the mutable address of the i-th matrix component without performing bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, dereferencing the result will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-07-09 00:12:43 +08:00
|
|
|
fn get_address_unchecked_linear_mut(&mut self, i: usize) -> *mut T {
|
2020-11-19 18:56:58 +08:00
|
|
|
self.ptr_mut().wrapping_add(i)
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Gets the mutable address of the i-th matrix component without performing bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, dereferencing the result will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-07-09 00:12:43 +08:00
|
|
|
fn get_address_unchecked_mut(&mut self, irow: usize, icol: usize) -> *mut T {
|
2016-12-05 05:44:42 +08:00
|
|
|
let lid = self.linear_index(irow, icol);
|
|
|
|
self.get_address_unchecked_linear_mut(lid)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves a mutable reference to the i-th element without bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, the method will cause undefined behavior.
|
2021-04-11 17:00:38 +08:00
|
|
|
unsafe fn get_unchecked_linear_mut(&mut self, i: usize) -> &mut T {
|
2016-12-05 05:44:42 +08:00
|
|
|
&mut *self.get_address_unchecked_linear_mut(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves a mutable reference to the element at `(irow, icol)` without bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the index is out of bounds, the method will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
unsafe fn get_unchecked_mut(&mut self, irow: usize, icol: usize) -> &mut T {
|
2016-12-05 05:44:42 +08:00
|
|
|
&mut *self.get_address_unchecked_mut(irow, icol)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Swaps two elements using their linear index without bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the indices are out of bounds, the method will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
|
|
|
unsafe fn swap_unchecked_linear(&mut self, i1: usize, i2: usize) {
|
|
|
|
let a = self.get_address_unchecked_linear_mut(i1);
|
|
|
|
let b = self.get_address_unchecked_linear_mut(i2);
|
|
|
|
|
2021-06-18 08:51:00 +08:00
|
|
|
ptr::swap(a, b);
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Swaps two elements without bound-checking.
|
2021-07-09 00:12:43 +08:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
/// If the indices are out of bounds, the method will cause undefined behavior.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
|
|
|
unsafe fn swap_unchecked(&mut self, row_col1: (usize, usize), row_col2: (usize, usize)) {
|
|
|
|
let lid1 = self.linear_index(row_col1.0, row_col1.1);
|
|
|
|
let lid2 = self.linear_index(row_col2.0, row_col2.1);
|
|
|
|
|
|
|
|
self.swap_unchecked_linear(lid1, lid2)
|
|
|
|
}
|
2017-08-03 01:37:44 +08:00
|
|
|
|
|
|
|
/// Retrieves the mutable data buffer as a contiguous slice.
|
|
|
|
///
|
2021-06-17 15:46:49 +08:00
|
|
|
/// Matrix components may not be contiguous, depending on its strides.
|
|
|
|
///
|
2021-07-06 05:51:27 +08:00
|
|
|
/// # Safety
|
2021-06-17 15:46:49 +08:00
|
|
|
/// The matrix components may not be stored in a contiguous way, depending on the strides.
|
|
|
|
/// This method is unsafe because this can yield to invalid aliasing when called on some pairs
|
|
|
|
/// of matrix slices originating from the same matrix with strides.
|
|
|
|
unsafe fn as_mut_slice_unchecked(&mut self) -> &mut [T];
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2021-08-03 00:41:46 +08:00
|
|
|
pub unsafe trait StorageMut<T, R: Dim, C: Dim = U1>:
|
|
|
|
Storage<T, R, C> + RawStorageMut<T, R, C>
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe impl<S, T, R, C> StorageMut<T, R, C> for S
|
|
|
|
where
|
|
|
|
R: Dim,
|
|
|
|
C: Dim,
|
|
|
|
S: Storage<T, R, C> + RawStorageMut<T, R, C>,
|
|
|
|
{
|
2018-02-02 19:26:35 +08:00
|
|
|
}
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2021-08-03 00:41:46 +08:00
|
|
|
/// Marker trait indicating that a storage is stored contiguously in memory.
|
2017-08-03 01:37:44 +08:00
|
|
|
///
|
2020-10-25 23:02:31 +08:00
|
|
|
/// The storage requirement means that for any value of `i` in `[0, nrows * ncols - 1]`, the value
|
2017-08-03 01:37:44 +08:00
|
|
|
/// `.get_unchecked_linear` returns one of the matrix component. This trait is unsafe because
|
|
|
|
/// failing to comply to this may cause Undefined Behaviors.
|
2021-08-03 00:41:46 +08:00
|
|
|
pub unsafe trait IsContiguous {}
|
2020-08-19 13:52:26 +08:00
|
|
|
|
|
|
|
/// A matrix storage that can be reshaped in-place.
|
2021-08-03 00:41:46 +08:00
|
|
|
pub trait ReshapableStorage<T, R1, C1, R2, C2>: RawStorage<T, R1, C1>
|
|
|
|
where
|
|
|
|
T: Scalar,
|
|
|
|
R1: Dim,
|
|
|
|
C1: Dim,
|
|
|
|
R2: Dim,
|
|
|
|
C2: Dim,
|
|
|
|
{
|
2020-08-19 13:52:26 +08:00
|
|
|
/// The reshaped storage type.
|
2021-08-03 00:41:46 +08:00
|
|
|
type Output: RawStorage<T, R2, C2>;
|
2020-08-19 13:52:26 +08:00
|
|
|
|
|
|
|
/// Reshapes the storage into the output storage type.
|
|
|
|
fn reshape_generic(self, nrows: R2, ncols: C2) -> Self::Output;
|
|
|
|
}
|