From 8555c995fcc8db8e1f078f93385c6c624fed47d0 Mon Sep 17 00:00:00 2001 From: Paul-Edouard Sarlin Date: Fri, 17 May 2024 18:07:06 +0200 Subject: [PATCH 1/3] Add NormalError cost function --- _pyceres/factors/bindings.h | 45 ++++++++++++++++++++++++++++++++++--- 1 file changed, 42 insertions(+), 3 deletions(-) diff --git a/_pyceres/factors/bindings.h b/_pyceres/factors/bindings.h index e2aca1d..b9c20a5 100644 --- a/_pyceres/factors/bindings.h +++ b/_pyceres/factors/bindings.h @@ -9,16 +9,55 @@ namespace py = pybind11; +inline Eigen::MatrixXd SqrtInformation(const Eigen::MatrixXd& covariance) { + return covariance.inverse().llt().matrixL(); +} + +class NormalError { + public: + explicit NormalError(const Eigen::MatrixXd& covariance) + : sqrt_information_(SqrtInformation(covariance)), + dimension_(covariance.rows()) { + THROW_CHECK_EQ(covariance.rows(), covariance.cols()); + } + + static ceres::CostFunction* Create(const Eigen::MatrixXd& covariance) { + auto* cost_function = new ceres::DynamicAutoDiffCostFunction( + new NormalError(covariance)); + const int dimension = covariance.rows(); + cost_function->AddParameterBlock(dimension); + cost_function->AddParameterBlock(dimension); + cost_function->SetNumResiduals(dimension); + return cost_function; + } + + template + bool operator()(T const* const* parameters, T* residuals_ptr) const { + for (int i = 0; i < dimension_; ++i) { + residuals_ptr[i] = parameters[0][i] - parameters[1][i]; + } + Eigen::Map> residuals(residuals_ptr, + dimension_); + residuals.applyOnTheLeft(sqrt_information_.template cast()); + return true; + } + + private: + const Eigen::MatrixXd sqrt_information_; + const int dimension_; +}; + void BindFactors(py::module& m) { m.def( "NormalPrior", [](const Eigen::VectorXd& mean, - const Eigen::Matrix& covariance) { + const Eigen::MatrixXd& covariance) -> ceres::CostFunction* { THROW_CHECK_EQ(covariance.cols(), mean.size()); THROW_CHECK_EQ(covariance.cols(), covariance.rows()); - return new ceres::NormalPrior(covariance.inverse().llt().matrixL(), - mean); + return new ceres::NormalPrior(SqrtInformation(covariance), mean); }, py::arg("mean"), py::arg("covariance")); + + m.def("NormalError", &NormalError::Create, py::arg("covariance")); } From af271c480bef4de20dc6a1c5c7e4afb1131ad818 Mon Sep 17 00:00:00 2001 From: Paul-Edouard Sarlin Date: Fri, 17 May 2024 18:08:38 +0200 Subject: [PATCH 2/3] Add comment --- _pyceres/factors/bindings.h | 1 + 1 file changed, 1 insertion(+) diff --git a/_pyceres/factors/bindings.h b/_pyceres/factors/bindings.h index b9c20a5..2da2ae9 100644 --- a/_pyceres/factors/bindings.h +++ b/_pyceres/factors/bindings.h @@ -13,6 +13,7 @@ inline Eigen::MatrixXd SqrtInformation(const Eigen::MatrixXd& covariance) { return covariance.inverse().llt().matrixL(); } +// Mahalanobis squared distance between two parameters. class NormalError { public: explicit NormalError(const Eigen::MatrixXd& covariance) From fb94a2509d448cbda4353a453d7494105d5038c3 Mon Sep 17 00:00:00 2001 From: Paul-Edouard Sarlin Date: Fri, 17 May 2024 18:17:24 +0200 Subject: [PATCH 3/3] Fix --- _pyceres/factors/bindings.h | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/_pyceres/factors/bindings.h b/_pyceres/factors/bindings.h index 2da2ae9..3601b96 100644 --- a/_pyceres/factors/bindings.h +++ b/_pyceres/factors/bindings.h @@ -17,8 +17,7 @@ inline Eigen::MatrixXd SqrtInformation(const Eigen::MatrixXd& covariance) { class NormalError { public: explicit NormalError(const Eigen::MatrixXd& covariance) - : sqrt_information_(SqrtInformation(covariance)), - dimension_(covariance.rows()) { + : sqrt_information_(SqrtInformation(covariance)) { THROW_CHECK_EQ(covariance.rows(), covariance.cols()); } @@ -34,18 +33,18 @@ class NormalError { template bool operator()(T const* const* parameters, T* residuals_ptr) const { - for (int i = 0; i < dimension_; ++i) { + const int dimension = sqrt_information_.rows(); + for (int i = 0; i < dimension; ++i) { residuals_ptr[i] = parameters[0][i] - parameters[1][i]; } Eigen::Map> residuals(residuals_ptr, - dimension_); + dimension); residuals.applyOnTheLeft(sqrt_information_.template cast()); return true; } private: const Eigen::MatrixXd sqrt_information_; - const int dimension_; }; void BindFactors(py::module& m) {