use core::num::{One, Zero}; use core::vec::{from_elem, swap, all, all2, len}; use core::cmp::ApproxEq; use traits::inv::Inv; use traits::transpose::Transpose; use traits::workarounds::rlmul::{RMul, LMul}; use ndim::dvec::{DVec, zero_vec_with_dim}; #[deriving(Eq, ToStr, Clone)] pub struct DMat { dim: uint, // FIXME: handle more than just square matrices mij: ~[T] } pub fn zero_mat_with_dim(dim: uint) -> DMat { DMat { dim: dim, mij: from_elem(dim * dim, Zero::zero()) } } pub fn is_zero_mat(mat: &DMat) -> bool { all(mat.mij, |e| e.is_zero()) } pub fn one_mat_with_dim(dim: uint) -> DMat { let mut res = zero_mat_with_dim(dim); let _1 = One::one::(); for uint::range(0u, dim) |i| { res.set(i, i, &_1); } res } impl DMat { pub fn offset(&self, i: uint, j: uint) -> uint { i * self.dim + j } pub fn set(&mut self, i: uint, j: uint, t: &T) { assert!(i < self.dim); assert!(j < self.dim); self.mij[self.offset(i, j)] = *t } } impl Index<(uint, uint), T> for DMat { fn index(&self, &(i, j): &(uint, uint)) -> T { self.mij[self.offset(i, j)] } } impl + Add + Zero> Mul, DMat> for DMat { fn mul(&self, other: &DMat) -> DMat { assert!(self.dim == other.dim); let dim = self.dim; let mut res = zero_mat_with_dim(dim); for uint::range(0u, dim) |i| { for uint::range(0u, dim) |j| { let mut acc: T = Zero::zero(); for uint::range(0u, dim) |k| { acc += self[(i, k)] * other[(k, j)]; } res.set(i, j, &acc); } } res } } impl + Mul + Zero> RMul> for DMat { fn rmul(&self, other: &DVec) -> DVec { assert!(self.dim == len(other.at)); let dim = self.dim; let mut res : DVec = zero_vec_with_dim(dim); for uint::range(0u, dim) |i| { for uint::range(0u, dim) |j| { res.at[i] = res.at[i] + other.at[j] * self[(i, j)]; } } res } } impl + Mul + Zero> LMul> for DMat { fn lmul(&self, other: &DVec) -> DVec { assert!(self.dim == len(other.at)); let dim = self.dim; let mut res : DVec = zero_vec_with_dim(dim); for uint::range(0u, dim) |i| { for uint::range(0u, dim) |j| { res.at[i] = res.at[i] + other.at[j] * self[(j, i)]; } } res } } impl + Div + Sub + Neg> Inv for DMat { fn inverse(&self) -> DMat { let mut res : DMat = self.clone(); res.invert(); res } fn invert(&mut self) { let dim = self.dim; let mut res = one_mat_with_dim::(dim); let _0T = Zero::zero::(); // inversion using Gauss-Jordan elimination for uint::range(0u, dim) |k| { // search a non-zero value on the k-th column // FIXME: would it be worth it to spend some more time searching for the // max instead? // FIXME: this is kind of uggly… // … but we cannot use position_between since we are iterating on one // columns let mut n0 = 0u; // index of a non-zero entry while (n0 != dim) { if (self[(n0, k)] != _0T) { break; } n0 += 1; } assert!(n0 != dim); // non inversible matrix // swap pivot line if (n0 != k) { for uint::range(0u, dim) |j| { let off_n0_j = self.offset(n0, j); let off_k_j = self.offset(k, j); swap(self.mij, off_n0_j, off_k_j); swap(res.mij, off_n0_j, off_k_j); } } let pivot = self[(k, k)]; for uint::range(k, dim) |j| { // FIXME: not to putting selfal exression directly on the nuction call // is uggly but does not seem to compile any more… let selfval = &(self[(k, j)] / pivot); let resval = &(res[(k, j)] / pivot); self.set(k, j, selfval); res.set(k, j, resval); } for uint::range(0u, dim) |l| { if (l != k) { let normalizer = self[(l, k)] / pivot; for uint::range(k, dim) |j| { let selfval = &(self[(l, j)] - self[(k, j)] * normalizer); let resval = &(res[(l, j)] - res[(k, j)] * normalizer); self.set(k, j, selfval); res.set(k, j, resval); } } } } } } impl Transpose for DMat { fn transposed(&self) -> DMat { let mut res = copy *self; res.transpose(); res } fn transpose(&mut self) { let dim = self.dim; for uint::range(1u, dim) |i| { for uint::range(0u, dim - 1) |j| { let off_i_j = self.offset(i, j); let off_j_i = self.offset(j, i); swap(self.mij, off_i_j, off_j_i); } } } } impl> ApproxEq for DMat { fn approx_epsilon() -> T { ApproxEq::approx_epsilon::() } fn approx_eq(&self, other: &DMat) -> bool { all2(self.mij, other.mij, |a, b| a.approx_eq(b)) } fn approx_eq_eps(&self, other: &DMat, epsilon: &T) -> bool { all2(self.mij, other.mij, |a, b| a.approx_eq_eps(b, epsilon)) } }