faer 0.24.0

linear algebra library
Documentation
use crate::assert;
use crate::internal_prelude::*;
pub fn solve_in_place_scratch<T: ComplexField>(
	dim: usize,
	rhs_ncols: usize,
	par: Par,
) -> StackReq {
	_ = (dim, rhs_ncols, par);
	StackReq::EMPTY
}
#[track_caller]
pub fn solve_in_place_with_conj<T: ComplexField>(
	L: MatRef<'_, T>,
	conj_lhs: Conj,
	rhs: MatMut<'_, T>,
	par: Par,
	stack: &mut MemStack,
) {
	let n = L.nrows();
	assert!(all(L.nrows() == n, L.ncols() == n, rhs.nrows() == n));
	_ = stack;
	let mut rhs = rhs;
	linalg::triangular_solve::solve_lower_triangular_in_place_with_conj(
		L,
		conj_lhs,
		rhs.rb_mut(),
		par,
	);
	linalg::triangular_solve::solve_upper_triangular_in_place_with_conj(
		L.transpose(),
		conj_lhs.compose(Conj::Yes),
		rhs.rb_mut(),
		par,
	);
}
#[track_caller]
pub fn solve_in_place<T: ComplexField, C: Conjugate<Canonical = T>>(
	L: MatRef<'_, C>,
	rhs: MatMut<'_, T>,
	par: Par,
	stack: &mut MemStack,
) {
	solve_in_place_with_conj(L.canonical(), Conj::get::<C>(), rhs, par, stack);
}
#[cfg(test)]
mod tests {
	use super::*;
	use crate::assert;
	use crate::linalg::cholesky::llt::factor::LltParams;
	use crate::stats::prelude::*;
	use crate::utils::approx::*;
	use dyn_stack::MemBuffer;
	use linalg::cholesky::llt;
	#[test]
	fn test_solve() {
		let rng = &mut StdRng::seed_from_u64(0);
		for n in [50, 200, 400] {
			let k = 3;
			let A = CwiseMatDistribution {
				nrows: n,
				ncols: n,
				dist: ComplexDistribution::new(StandardNormal, StandardNormal),
			}
			.rand::<Mat<c64>>(rng);
			let B = CwiseMatDistribution {
				nrows: n,
				ncols: k,
				dist: ComplexDistribution::new(StandardNormal, StandardNormal),
			}
			.rand::<Mat<c64>>(rng);
			let A = &A * A.adjoint();
			let mut L = A.to_owned();
			llt::factor::cholesky_in_place(
				L.as_mut(),
				Default::default(),
				Par::Seq,
				MemStack::new(&mut {
					MemBuffer::new(
						llt::factor::cholesky_in_place_scratch::<c64>(
							n,
							Par::Seq,
							LltParams {
								recursion_threshold: 32,
								block_size: 128,
								..auto!(c64)
							}
							.into(),
						),
					)
				}),
				LltParams {
					recursion_threshold: 32,
					block_size: 128,
					..auto!(c64)
				}
				.into(),
			)
			.unwrap();
			let approx_eq = CwiseMat(ApproxEq::eps() * 8.0 * (n as f64));
			{
				let mut X = B.to_owned();
				llt::solve::solve_in_place(
					L.as_ref(),
					X.as_mut(),
					Par::Seq,
					MemStack::new(&mut MemBuffer::new(
						llt::solve::solve_in_place_scratch::<c64>(
							n,
							k,
							Par::Seq,
						),
					)),
				);
				assert!(& A * & X ~ B);
			}
			{
				let mut X = B.to_owned();
				llt::solve::solve_in_place(
					L.conjugate(),
					X.as_mut(),
					Par::Seq,
					MemStack::new(&mut MemBuffer::new(
						llt::solve::solve_in_place_scratch::<c64>(
							n,
							k,
							Par::Seq,
						),
					)),
				);
				assert!(A.conjugate() * & X ~ B);
			}
		}
	}
}