2019-03-23 18:46:56 +08:00
|
|
|
|
use num::Zero;
|
2020-03-21 19:16:46 +08:00
|
|
|
|
use std::ops::Neg;
|
2018-12-09 18:21:24 +08:00
|
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
|
use crate::allocator::Allocator;
|
2020-03-25 02:06:05 +08:00
|
|
|
|
use crate::base::{DefaultAllocator, Dim, DimName, Matrix, MatrixMN, Normed, VectorN};
|
2020-03-18 00:58:36 +08:00
|
|
|
|
use crate::constraint::{SameNumberOfColumns, SameNumberOfRows, ShapeConstraint};
|
2019-03-23 21:29:07 +08:00
|
|
|
|
use crate::storage::{Storage, StorageMut};
|
2020-03-21 19:16:46 +08:00
|
|
|
|
use crate::{ComplexField, Scalar, SimdComplexField, Unit};
|
|
|
|
|
use simba::scalar::ClosedNeg;
|
|
|
|
|
use simba::simd::{SimdOption, SimdPartialOrd};
|
2018-12-09 18:21:24 +08:00
|
|
|
|
|
|
|
|
|
// FIXME: this should be be a trait on alga?
|
2019-02-03 15:33:07 +08:00
|
|
|
|
/// A trait for abstract matrix norms.
|
|
|
|
|
///
|
|
|
|
|
/// This may be moved to the alga crate in the future.
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub trait Norm<N: SimdComplexField> {
|
2019-02-03 15:33:07 +08:00
|
|
|
|
/// Apply this norm to the given matrix.
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn norm<R, C, S>(&self, m: &Matrix<N, R, C, S>) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R: Dim,
|
|
|
|
|
C: Dim,
|
|
|
|
|
S: Storage<N, R, C>;
|
2019-02-03 15:33:07 +08:00
|
|
|
|
/// Use the metric induced by this norm to compute the metric distance between the two given matrices.
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn metric_distance<R1, C1, S1, R2, C2, S2>(
|
|
|
|
|
&self,
|
|
|
|
|
m1: &Matrix<N, R1, C1, S1>,
|
|
|
|
|
m2: &Matrix<N, R2, C2, S2>,
|
|
|
|
|
) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R1: Dim,
|
|
|
|
|
C1: Dim,
|
|
|
|
|
S1: Storage<N, R1, C1>,
|
|
|
|
|
R2: Dim,
|
|
|
|
|
C2: Dim,
|
|
|
|
|
S2: Storage<N, R2, C2>,
|
|
|
|
|
ShapeConstraint: SameNumberOfRows<R1, R2> + SameNumberOfColumns<C1, C2>;
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Euclidean norm.
|
|
|
|
|
pub struct EuclideanNorm;
|
|
|
|
|
/// Lp norm.
|
|
|
|
|
pub struct LpNorm(pub i32);
|
|
|
|
|
/// L-infinite norm aka. Chebytchev norm aka. uniform norm aka. suppremum norm.
|
|
|
|
|
pub struct UniformNorm;
|
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
impl<N: SimdComplexField> Norm<N> for EuclideanNorm {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn norm<R, C, S>(&self, m: &Matrix<N, R, C, S>) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R: Dim,
|
|
|
|
|
C: Dim,
|
|
|
|
|
S: Storage<N, R, C>,
|
|
|
|
|
{
|
|
|
|
|
m.norm_squared().simd_sqrt()
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn metric_distance<R1, C1, S1, R2, C2, S2>(
|
|
|
|
|
&self,
|
|
|
|
|
m1: &Matrix<N, R1, C1, S1>,
|
|
|
|
|
m2: &Matrix<N, R2, C2, S2>,
|
|
|
|
|
) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R1: Dim,
|
|
|
|
|
C1: Dim,
|
|
|
|
|
S1: Storage<N, R1, C1>,
|
|
|
|
|
R2: Dim,
|
|
|
|
|
C2: Dim,
|
|
|
|
|
S2: Storage<N, R2, C2>,
|
|
|
|
|
ShapeConstraint: SameNumberOfRows<R1, R2> + SameNumberOfColumns<C1, C2>,
|
|
|
|
|
{
|
|
|
|
|
m1.zip_fold(m2, N::SimdRealField::zero(), |acc, a, b| {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
let diff = a - b;
|
2020-03-18 00:58:36 +08:00
|
|
|
|
acc + diff.simd_modulus_squared()
|
|
|
|
|
})
|
|
|
|
|
.simd_sqrt()
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
impl<N: SimdComplexField> Norm<N> for LpNorm {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn norm<R, C, S>(&self, m: &Matrix<N, R, C, S>) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R: Dim,
|
|
|
|
|
C: Dim,
|
|
|
|
|
S: Storage<N, R, C>,
|
|
|
|
|
{
|
|
|
|
|
m.fold(N::SimdRealField::zero(), |a, b| {
|
|
|
|
|
a + b.simd_modulus().simd_powi(self.0)
|
|
|
|
|
})
|
|
|
|
|
.simd_powf(crate::convert(1.0 / (self.0 as f64)))
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn metric_distance<R1, C1, S1, R2, C2, S2>(
|
|
|
|
|
&self,
|
|
|
|
|
m1: &Matrix<N, R1, C1, S1>,
|
|
|
|
|
m2: &Matrix<N, R2, C2, S2>,
|
|
|
|
|
) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R1: Dim,
|
|
|
|
|
C1: Dim,
|
|
|
|
|
S1: Storage<N, R1, C1>,
|
|
|
|
|
R2: Dim,
|
|
|
|
|
C2: Dim,
|
|
|
|
|
S2: Storage<N, R2, C2>,
|
|
|
|
|
ShapeConstraint: SameNumberOfRows<R1, R2> + SameNumberOfColumns<C1, C2>,
|
|
|
|
|
{
|
|
|
|
|
m1.zip_fold(m2, N::SimdRealField::zero(), |acc, a, b| {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
let diff = a - b;
|
2020-03-18 00:58:36 +08:00
|
|
|
|
acc + diff.simd_modulus().simd_powi(self.0)
|
|
|
|
|
})
|
|
|
|
|
.simd_powf(crate::convert(1.0 / (self.0 as f64)))
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
impl<N: SimdComplexField> Norm<N> for UniformNorm {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn norm<R, C, S>(&self, m: &Matrix<N, R, C, S>) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R: Dim,
|
|
|
|
|
C: Dim,
|
|
|
|
|
S: Storage<N, R, C>,
|
|
|
|
|
{
|
2019-03-23 21:13:00 +08:00
|
|
|
|
// NOTE: we don't use `m.amax()` here because for the complex
|
|
|
|
|
// numbers this will return the max norm1 instead of the modulus.
|
2020-03-18 00:58:36 +08:00
|
|
|
|
m.fold(N::SimdRealField::zero(), |acc, a| {
|
|
|
|
|
acc.simd_max(a.simd_modulus())
|
|
|
|
|
})
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
fn metric_distance<R1, C1, S1, R2, C2, S2>(
|
|
|
|
|
&self,
|
|
|
|
|
m1: &Matrix<N, R1, C1, S1>,
|
|
|
|
|
m2: &Matrix<N, R2, C2, S2>,
|
|
|
|
|
) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R1: Dim,
|
|
|
|
|
C1: Dim,
|
|
|
|
|
S1: Storage<N, R1, C1>,
|
|
|
|
|
R2: Dim,
|
|
|
|
|
C2: Dim,
|
|
|
|
|
S2: Storage<N, R2, C2>,
|
|
|
|
|
ShapeConstraint: SameNumberOfRows<R1, R2> + SameNumberOfColumns<C1, C2>,
|
|
|
|
|
{
|
|
|
|
|
m1.zip_fold(m2, N::SimdRealField::zero(), |acc, a, b| {
|
|
|
|
|
let val = (a - b).simd_modulus();
|
|
|
|
|
acc.simd_max(val)
|
2018-12-09 18:21:24 +08:00
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
impl<N: SimdComplexField, R: Dim, C: Dim, S: Storage<N, R, C>> Matrix<N, R, C, S> {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
/// The squared L2 norm of this vector.
|
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn norm_squared(&self) -> N::SimdRealField {
|
|
|
|
|
let mut res = N::SimdRealField::zero();
|
2018-12-09 18:21:24 +08:00
|
|
|
|
|
|
|
|
|
for i in 0..self.ncols() {
|
|
|
|
|
let col = self.column(i);
|
2020-03-18 00:58:36 +08:00
|
|
|
|
res += col.dotc(&col).simd_real()
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
res
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The L2 norm of this matrix.
|
2019-02-03 15:33:07 +08:00
|
|
|
|
///
|
|
|
|
|
/// Use `.apply_norm` to apply a custom norm.
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn norm(&self) -> N::SimdRealField {
|
|
|
|
|
self.norm_squared().simd_sqrt()
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-03 15:33:07 +08:00
|
|
|
|
/// Compute the distance between `self` and `rhs` using the metric induced by the euclidean norm.
|
|
|
|
|
///
|
|
|
|
|
/// Use `.apply_metric_distance` to apply a custom norm.
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn metric_distance<R2, C2, S2>(&self, rhs: &Matrix<N, R2, C2, S2>) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R2: Dim,
|
|
|
|
|
C2: Dim,
|
|
|
|
|
S2: Storage<N, R2, C2>,
|
|
|
|
|
ShapeConstraint: SameNumberOfRows<R, R2> + SameNumberOfColumns<C, C2>,
|
|
|
|
|
{
|
2018-12-09 18:21:24 +08:00
|
|
|
|
self.apply_metric_distance(rhs, &EuclideanNorm)
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-03 15:33:07 +08:00
|
|
|
|
/// Uses the given `norm` to compute the norm of `self`.
|
2019-02-03 18:29:10 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// # use nalgebra::{Vector3, UniformNorm, LpNorm, EuclideanNorm};
|
|
|
|
|
///
|
|
|
|
|
/// let v = Vector3::new(1.0, 2.0, 3.0);
|
|
|
|
|
/// assert_eq!(v.apply_norm(&UniformNorm), 3.0);
|
|
|
|
|
/// assert_eq!(v.apply_norm(&LpNorm(1)), 6.0);
|
|
|
|
|
/// assert_eq!(v.apply_norm(&EuclideanNorm), v.norm());
|
|
|
|
|
/// ```
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn apply_norm(&self, norm: &impl Norm<N>) -> N::SimdRealField {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
norm.norm(self)
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-03 15:33:07 +08:00
|
|
|
|
/// Uses the metric induced by the given `norm` to compute the metric distance between `self` and `rhs`.
|
2019-02-03 18:29:10 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// # use nalgebra::{Vector3, UniformNorm, LpNorm, EuclideanNorm};
|
|
|
|
|
///
|
|
|
|
|
/// let v1 = Vector3::new(1.0, 2.0, 3.0);
|
|
|
|
|
/// let v2 = Vector3::new(10.0, 20.0, 30.0);
|
|
|
|
|
///
|
|
|
|
|
/// assert_eq!(v1.apply_metric_distance(&v2, &UniformNorm), 27.0);
|
|
|
|
|
/// assert_eq!(v1.apply_metric_distance(&v2, &LpNorm(1)), 27.0 + 18.0 + 9.0);
|
|
|
|
|
/// assert_eq!(v1.apply_metric_distance(&v2, &EuclideanNorm), (v1 - v2).norm());
|
|
|
|
|
/// ```
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn apply_metric_distance<R2, C2, S2>(
|
|
|
|
|
&self,
|
|
|
|
|
rhs: &Matrix<N, R2, C2, S2>,
|
|
|
|
|
norm: &impl Norm<N>,
|
|
|
|
|
) -> N::SimdRealField
|
|
|
|
|
where
|
|
|
|
|
R2: Dim,
|
|
|
|
|
C2: Dim,
|
|
|
|
|
S2: Storage<N, R2, C2>,
|
|
|
|
|
ShapeConstraint: SameNumberOfRows<R, R2> + SameNumberOfColumns<C, C2>,
|
|
|
|
|
{
|
2019-02-23 18:24:07 +08:00
|
|
|
|
norm.metric_distance(self, rhs)
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A synonym for the norm of this matrix.
|
|
|
|
|
///
|
|
|
|
|
/// Aka the length.
|
|
|
|
|
///
|
|
|
|
|
/// This function is simply implemented as a call to `norm()`
|
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn magnitude(&self) -> N::SimdRealField {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
self.norm()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A synonym for the squared norm of this matrix.
|
|
|
|
|
///
|
|
|
|
|
/// Aka the squared length.
|
|
|
|
|
///
|
|
|
|
|
/// This function is simply implemented as a call to `norm_squared()`
|
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn magnitude_squared(&self) -> N::SimdRealField {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
self.norm_squared()
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
/// Sets the magnitude of this vector.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn set_magnitude(&mut self, magnitude: N::SimdRealField)
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
S: StorageMut<N, R, C>,
|
|
|
|
|
{
|
2020-03-18 00:58:36 +08:00
|
|
|
|
let n = self.norm();
|
|
|
|
|
self.scale_mut(magnitude / n)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns a normalized version of this matrix.
|
|
|
|
|
#[inline]
|
|
|
|
|
#[must_use = "Did you mean to use normalize_mut()?"]
|
|
|
|
|
pub fn normalize(&self) -> MatrixMN<N, R, C>
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
DefaultAllocator: Allocator<N, R, C>,
|
|
|
|
|
{
|
2020-03-18 00:58:36 +08:00
|
|
|
|
self.unscale(self.norm())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The Lp norm of this matrix.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn lp_norm(&self, p: i32) -> N::SimdRealField {
|
|
|
|
|
self.apply_norm(&LpNorm(p))
|
|
|
|
|
}
|
2020-03-21 19:16:46 +08:00
|
|
|
|
|
2020-04-06 00:02:03 +08:00
|
|
|
|
/// Attempts to normalize `self`.
|
|
|
|
|
///
|
|
|
|
|
/// The components of this matrix can be SIMD types.
|
2020-03-21 19:16:46 +08:00
|
|
|
|
#[inline]
|
|
|
|
|
#[must_use = "Did you mean to use simd_try_normalize_mut()?"]
|
|
|
|
|
pub fn simd_try_normalize(&self, min_norm: N::SimdRealField) -> SimdOption<MatrixMN<N, R, C>>
|
|
|
|
|
where
|
|
|
|
|
N::Element: Scalar,
|
|
|
|
|
DefaultAllocator: Allocator<N, R, C> + Allocator<N::Element, R, C>,
|
|
|
|
|
{
|
|
|
|
|
let n = self.norm();
|
|
|
|
|
let le = n.simd_le(min_norm);
|
|
|
|
|
let val = self.unscale(n);
|
|
|
|
|
SimdOption::new(val, le)
|
|
|
|
|
}
|
2020-03-18 00:58:36 +08:00
|
|
|
|
}
|
2020-01-01 22:59:46 +08:00
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
impl<N: ComplexField, R: Dim, C: Dim, S: Storage<N, R, C>> Matrix<N, R, C, S> {
|
2020-01-01 22:59:46 +08:00
|
|
|
|
/// Sets the magnitude of this vector unless it is smaller than `min_magnitude`.
|
|
|
|
|
///
|
|
|
|
|
/// If `self.magnitude()` is smaller than `min_magnitude`, it will be left unchanged.
|
|
|
|
|
/// Otherwise this is equivalent to: `*self = self.normalize() * magnitude.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn try_set_magnitude(&mut self, magnitude: N::RealField, min_magnitude: N::RealField)
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
S: StorageMut<N, R, C>,
|
|
|
|
|
{
|
2020-01-01 22:59:46 +08:00
|
|
|
|
let n = self.norm();
|
|
|
|
|
|
|
|
|
|
if n >= min_magnitude {
|
|
|
|
|
self.scale_mut(magnitude / n)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-09 18:21:24 +08:00
|
|
|
|
/// Returns a normalized version of this matrix unless its norm as smaller or equal to `eps`.
|
2020-04-06 00:02:03 +08:00
|
|
|
|
///
|
|
|
|
|
/// The components of this matrix cannot be SIMD types (see `simd_try_normalize`) instead.
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2019-06-06 05:04:04 +08:00
|
|
|
|
#[must_use = "Did you mean to use try_normalize_mut()?"]
|
2019-03-25 18:21:41 +08:00
|
|
|
|
pub fn try_normalize(&self, min_norm: N::RealField) -> Option<MatrixMN<N, R, C>>
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
DefaultAllocator: Allocator<N, R, C>,
|
|
|
|
|
{
|
2018-12-09 18:21:24 +08:00
|
|
|
|
let n = self.norm();
|
|
|
|
|
|
|
|
|
|
if n <= min_norm {
|
|
|
|
|
None
|
|
|
|
|
} else {
|
2019-03-23 21:13:00 +08:00
|
|
|
|
Some(self.unscale(n))
|
2018-12-09 18:21:24 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
impl<N: SimdComplexField, R: Dim, C: Dim, S: StorageMut<N, R, C>> Matrix<N, R, C, S> {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
/// Normalizes this matrix in-place and returns its norm.
|
2020-04-06 00:02:03 +08:00
|
|
|
|
///
|
|
|
|
|
/// The components of the matrix cannot be SIMD types (see `simd_try_normalize_mut` instead).
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-18 00:58:36 +08:00
|
|
|
|
pub fn normalize_mut(&mut self) -> N::SimdRealField {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
let n = self.norm();
|
2019-03-23 21:13:00 +08:00
|
|
|
|
self.unscale_mut(n);
|
2018-12-09 18:21:24 +08:00
|
|
|
|
|
|
|
|
|
n
|
|
|
|
|
}
|
2020-03-21 19:16:46 +08:00
|
|
|
|
|
2020-04-06 00:02:03 +08:00
|
|
|
|
/// Normalizes this matrix in-place and return its norm.
|
|
|
|
|
///
|
|
|
|
|
/// The components of the matrix can be SIMD types.
|
2020-03-21 19:16:46 +08:00
|
|
|
|
#[inline]
|
|
|
|
|
#[must_use = "Did you mean to use simd_try_normalize_mut()?"]
|
|
|
|
|
pub fn simd_try_normalize_mut(
|
|
|
|
|
&mut self,
|
|
|
|
|
min_norm: N::SimdRealField,
|
|
|
|
|
) -> SimdOption<N::SimdRealField>
|
|
|
|
|
where
|
|
|
|
|
N::Element: Scalar,
|
|
|
|
|
DefaultAllocator: Allocator<N, R, C> + Allocator<N::Element, R, C>,
|
|
|
|
|
{
|
|
|
|
|
let n = self.norm();
|
|
|
|
|
let le = n.simd_le(min_norm);
|
|
|
|
|
self.apply(|e| e.simd_unscale(n).select(le, e));
|
|
|
|
|
SimdOption::new(n, le)
|
|
|
|
|
}
|
2020-03-18 00:58:36 +08:00
|
|
|
|
}
|
2018-12-09 18:21:24 +08:00
|
|
|
|
|
2020-03-18 00:58:36 +08:00
|
|
|
|
impl<N: ComplexField, R: Dim, C: Dim, S: StorageMut<N, R, C>> Matrix<N, R, C, S> {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
/// Normalizes this matrix in-place or does nothing if its norm is smaller or equal to `eps`.
|
|
|
|
|
///
|
2020-01-01 22:59:46 +08:00
|
|
|
|
/// If the normalization succeeded, returns the old norm of this matrix.
|
2018-12-09 18:21:24 +08:00
|
|
|
|
#[inline]
|
2020-03-21 19:16:46 +08:00
|
|
|
|
pub fn try_normalize_mut(&mut self, min_norm: N::RealField) -> Option<N::RealField> {
|
2018-12-09 18:21:24 +08:00
|
|
|
|
let n = self.norm();
|
|
|
|
|
|
|
|
|
|
if n <= min_norm {
|
|
|
|
|
None
|
|
|
|
|
} else {
|
2019-03-23 21:13:00 +08:00
|
|
|
|
self.unscale_mut(n);
|
2018-12-09 18:21:24 +08:00
|
|
|
|
Some(n)
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-02-03 17:56:30 +08:00
|
|
|
|
}
|
2020-03-21 19:16:46 +08:00
|
|
|
|
|
|
|
|
|
impl<N: SimdComplexField, R: Dim, C: Dim> Normed for MatrixMN<N, R, C>
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
DefaultAllocator: Allocator<N, R, C>,
|
2020-03-21 19:16:46 +08:00
|
|
|
|
{
|
|
|
|
|
type Norm = N::SimdRealField;
|
|
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
|
fn norm(&self) -> N::SimdRealField {
|
|
|
|
|
self.norm()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
|
fn norm_squared(&self) -> N::SimdRealField {
|
|
|
|
|
self.norm_squared()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
|
fn scale_mut(&mut self, n: Self::Norm) {
|
|
|
|
|
self.scale_mut(n)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
|
fn unscale_mut(&mut self, n: Self::Norm) {
|
|
|
|
|
self.unscale_mut(n)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<N: Scalar + ClosedNeg, R: Dim, C: Dim> Neg for Unit<MatrixMN<N, R, C>>
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
DefaultAllocator: Allocator<N, R, C>,
|
2020-03-21 19:16:46 +08:00
|
|
|
|
{
|
|
|
|
|
type Output = Unit<MatrixMN<N, R, C>>;
|
|
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
|
fn neg(self) -> Self::Output {
|
|
|
|
|
Unit::new_unchecked(-self.value)
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-03-25 02:06:05 +08:00
|
|
|
|
|
|
|
|
|
// FIXME: specialization will greatly simplify this implementation in the future.
|
|
|
|
|
// In particular:
|
|
|
|
|
// − use `x()` instead of `::canonical_basis_element`
|
|
|
|
|
// − use `::new(x, y, z)` instead of `::from_slice`
|
|
|
|
|
impl<N: ComplexField, D: DimName> VectorN<N, D>
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
DefaultAllocator: Allocator<N, D>,
|
2020-03-25 02:06:05 +08:00
|
|
|
|
{
|
|
|
|
|
/// The i-the canonical basis element.
|
|
|
|
|
#[inline]
|
|
|
|
|
fn canonical_basis_element(i: usize) -> Self {
|
|
|
|
|
assert!(i < D::dim(), "Index out of bound.");
|
|
|
|
|
|
|
|
|
|
let mut res = Self::zero();
|
|
|
|
|
unsafe {
|
|
|
|
|
*res.data.get_unchecked_linear_mut(i) = N::one();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
res
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Orthonormalizes the given family of vectors. The largest free family of vectors is moved at
|
|
|
|
|
/// the beginning of the array and its size is returned. Vectors at an indices larger or equal to
|
|
|
|
|
/// this length can be modified to an arbitrary value.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn orthonormalize(vs: &mut [Self]) -> usize {
|
|
|
|
|
let mut nbasis_elements = 0;
|
|
|
|
|
|
|
|
|
|
for i in 0..vs.len() {
|
|
|
|
|
{
|
|
|
|
|
let (elt, basis) = vs[..i + 1].split_last_mut().unwrap();
|
|
|
|
|
|
|
|
|
|
for basis_element in &basis[..nbasis_elements] {
|
|
|
|
|
*elt -= &*basis_element * elt.dot(basis_element)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if vs[i].try_normalize_mut(N::RealField::zero()).is_some() {
|
|
|
|
|
// FIXME: this will be efficient on dynamically-allocated vectors but for
|
|
|
|
|
// statically-allocated ones, `.clone_from` would be better.
|
|
|
|
|
vs.swap(nbasis_elements, i);
|
|
|
|
|
nbasis_elements += 1;
|
|
|
|
|
|
|
|
|
|
// All the other vectors will be dependent.
|
|
|
|
|
if nbasis_elements == D::dim() {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nbasis_elements
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Applies the given closure to each element of the orthonormal basis of the subspace
|
|
|
|
|
/// orthogonal to free family of vectors `vs`. If `vs` is not a free family, the result is
|
|
|
|
|
/// unspecified.
|
|
|
|
|
// FIXME: return an iterator instead when `-> impl Iterator` will be supported by Rust.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn orthonormal_subspace_basis<F>(vs: &[Self], mut f: F)
|
2020-04-06 00:02:03 +08:00
|
|
|
|
where
|
|
|
|
|
F: FnMut(&Self) -> bool,
|
|
|
|
|
{
|
2020-03-25 02:06:05 +08:00
|
|
|
|
// FIXME: is this necessary?
|
|
|
|
|
assert!(
|
|
|
|
|
vs.len() <= D::dim(),
|
|
|
|
|
"The given set of vectors has no chance of being a free family."
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
match D::dim() {
|
|
|
|
|
1 => {
|
|
|
|
|
if vs.len() == 0 {
|
|
|
|
|
let _ = f(&Self::canonical_basis_element(0));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
2 => {
|
|
|
|
|
if vs.len() == 0 {
|
|
|
|
|
let _ = f(&Self::canonical_basis_element(0))
|
|
|
|
|
&& f(&Self::canonical_basis_element(1));
|
|
|
|
|
} else if vs.len() == 1 {
|
|
|
|
|
let v = &vs[0];
|
|
|
|
|
let res = Self::from_column_slice(&[-v[1], v[0]]);
|
|
|
|
|
|
|
|
|
|
let _ = f(&res.normalize());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Otherwise, nothing.
|
|
|
|
|
}
|
|
|
|
|
3 => {
|
|
|
|
|
if vs.len() == 0 {
|
|
|
|
|
let _ = f(&Self::canonical_basis_element(0))
|
|
|
|
|
&& f(&Self::canonical_basis_element(1))
|
|
|
|
|
&& f(&Self::canonical_basis_element(2));
|
|
|
|
|
} else if vs.len() == 1 {
|
|
|
|
|
let v = &vs[0];
|
|
|
|
|
let mut a;
|
|
|
|
|
|
|
|
|
|
if v[0].norm1() > v[1].norm1() {
|
|
|
|
|
a = Self::from_column_slice(&[v[2], N::zero(), -v[0]]);
|
|
|
|
|
} else {
|
|
|
|
|
a = Self::from_column_slice(&[N::zero(), -v[2], v[1]]);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let _ = a.normalize_mut();
|
|
|
|
|
|
|
|
|
|
if f(&a.cross(v)) {
|
|
|
|
|
let _ = f(&a);
|
|
|
|
|
}
|
|
|
|
|
} else if vs.len() == 2 {
|
|
|
|
|
let _ = f(&vs[0].cross(&vs[1]).normalize());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_ => {
|
|
|
|
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
|
|
|
|
{
|
|
|
|
|
// XXX: use a GenericArray instead.
|
|
|
|
|
let mut known_basis = Vec::new();
|
|
|
|
|
|
|
|
|
|
for v in vs.iter() {
|
|
|
|
|
known_basis.push(v.normalize())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for i in 0..D::dim() - vs.len() {
|
|
|
|
|
let mut elt = Self::canonical_basis_element(i);
|
|
|
|
|
|
|
|
|
|
for v in &known_basis {
|
|
|
|
|
elt -= v * elt.dot(v)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if let Some(subsp_elt) = elt.try_normalize(N::RealField::zero()) {
|
|
|
|
|
if !f(&subsp_elt) {
|
|
|
|
|
return;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
known_basis.push(subsp_elt);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#[cfg(all(not(feature = "std"), not(feature = "alloc")))]
|
|
|
|
|
{
|
|
|
|
|
panic!("Cannot compute the orthogonal subspace basis of a vector with a dimension greater than 3 \
|
|
|
|
|
if #![no_std] is enabled and the 'alloc' feature is not enabled.")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|