// SPDX-License-Identifier: Apache-2.0 // // Copyright 2008-2016 Conrad Sanderson (http://conradsanderson.id.au) // Copyright 2008-2016 National ICT Australia (NICTA) // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // ------------------------------------------------------------------------ //! \addtogroup op_det //! @{ template inline bool op_det::apply_direct(typename T1::elem_type& out_val, const Base& expr) { arma_extra_debug_sigprint(); typedef typename T1::elem_type eT; typedef typename T1::pod_type T; if(strip_diagmat::do_diagmat) { const strip_diagmat strip(expr.get_ref()); out_val = op_det::apply_diagmat(strip.M); return true; } if(strip_trimat::do_trimat) { const strip_trimat strip(expr.get_ref()); out_val = op_det::apply_trimat(strip.M); return true; } Mat A(expr.get_ref()); arma_debug_check( (A.is_square() == false), "det(): given matrix must be square sized" ); const uword N = A.n_rows; if(N == 0) { out_val = eT(1); return true; } if(N == 1) { out_val = A[0]; return true; } if((is_cx::no) && (N <= 3)) { constexpr T det_min = std::numeric_limits::epsilon(); constexpr T det_max = T(1) / std::numeric_limits::epsilon(); eT det_val = eT(0); if(N == 2) { det_val = op_det::apply_tiny_2x2(A); } if(N == 3) { det_val = op_det::apply_tiny_3x3(A); } const T abs_det_val = std::abs(det_val); if((abs_det_val > det_min) && (abs_det_val < det_max)) { out_val = det_val; return true; } // fallthrough if det_val is suspect } if(A.is_diagmat()) { out_val = op_det::apply_diagmat(A); return true; } const bool is_triu = trimat_helper::is_triu(A); const bool is_tril = is_triu ? false : trimat_helper::is_tril(A); if(is_triu || is_tril) { out_val = op_det::apply_trimat(A); return true; } return auxlib::det(out_val, A); } template inline typename T1::elem_type op_det::apply_diagmat(const Base& expr) { arma_extra_debug_sigprint(); typedef typename T1::elem_type eT; const diagmat_proxy A(expr.get_ref()); arma_debug_check( (A.n_rows != A.n_cols), "det(): given matrix must be square sized" ); const uword N = (std::min)(A.n_rows, A.n_cols); eT val = eT(1); for(uword i=0; i inline typename T1::elem_type op_det::apply_trimat(const Base& expr) { arma_extra_debug_sigprint(); typedef typename T1::elem_type eT; const Proxy P(expr.get_ref()); const uword N = P.get_n_rows(); arma_debug_check( (N != P.get_n_cols()), "det(): given matrix must be square sized" ); eT val = eT(1); for(uword i=0; i inline eT op_det::apply_tiny_2x2(const Mat& X) { arma_extra_debug_sigprint(); const eT* Xm = X.memptr(); return ( Xm[pos<0,0>::n2]*Xm[pos<1,1>::n2] - Xm[pos<0,1>::n2]*Xm[pos<1,0>::n2] ); } template inline eT op_det::apply_tiny_3x3(const Mat& X) { arma_extra_debug_sigprint(); const eT* Xm = X.memptr(); // const double tmp1 = X.at(0,0) * X.at(1,1) * X.at(2,2); // const double tmp2 = X.at(0,1) * X.at(1,2) * X.at(2,0); // const double tmp3 = X.at(0,2) * X.at(1,0) * X.at(2,1); // const double tmp4 = X.at(2,0) * X.at(1,1) * X.at(0,2); // const double tmp5 = X.at(2,1) * X.at(1,2) * X.at(0,0); // const double tmp6 = X.at(2,2) * X.at(1,0) * X.at(0,1); // return (tmp1+tmp2+tmp3) - (tmp4+tmp5+tmp6); const eT val1 = Xm[pos<0,0>::n3]*(Xm[pos<2,2>::n3]*Xm[pos<1,1>::n3] - Xm[pos<2,1>::n3]*Xm[pos<1,2>::n3]); const eT val2 = Xm[pos<1,0>::n3]*(Xm[pos<2,2>::n3]*Xm[pos<0,1>::n3] - Xm[pos<2,1>::n3]*Xm[pos<0,2>::n3]); const eT val3 = Xm[pos<2,0>::n3]*(Xm[pos<1,2>::n3]*Xm[pos<0,1>::n3] - Xm[pos<1,1>::n3]*Xm[pos<0,2>::n3]); return ( val1 - val2 + val3 ); } //! @}