diff --git a/_pyceres/factors/bindings.h b/_pyceres/factors/bindings.h index 3601b96..025cc70 100644 --- a/_pyceres/factors/bindings.h +++ b/_pyceres/factors/bindings.h @@ -10,7 +10,17 @@ namespace py = pybind11; inline Eigen::MatrixXd SqrtInformation(const Eigen::MatrixXd& covariance) { - return covariance.inverse().llt().matrixL(); + // LLT decomposition of Fisher information matrix with SVD + Eigen::JacobiSVD svd( + covariance, Eigen::ComputeThinU | Eigen::ComputeThinV); + Eigen::MatrixXd U = svd.matrixU(); + Eigen::VectorXd singularValues = svd.singularValues(); + const double epsilon = 1e-12; + Eigen::VectorXd invSqrtSingularValues = + singularValues.array().max(0.).sqrt().max(epsilon).inverse(); + Eigen::MatrixXd invSqrtSingularValuesDiag = + invSqrtSingularValues.asDiagonal(); + return U * invSqrtSingularValuesDiag; } // Mahalanobis squared distance between two parameters. @@ -39,7 +49,7 @@ class NormalError { } Eigen::Map> residuals(residuals_ptr, dimension); - residuals.applyOnTheLeft(sqrt_information_.template cast()); + residuals.applyOnTheLeft(sqrt_information_.transpose().template cast()); return true; }