From b84ea3aa5a5bd9d79cdeba201c22b6c05fc45662 Mon Sep 17 00:00:00 2001 From: "sewon.jeon" Date: Fri, 8 Dec 2023 18:55:20 +0900 Subject: [PATCH 1/2] symmetric icp --- .../registration/transformation_estimation.cu | 147 +++++++++++++++++- .../registration/transformation_estimation.h | 30 +++- .../registration/registration.cpp | 31 +++- 3 files changed, 200 insertions(+), 8 deletions(-) diff --git a/src/cupoch/registration/transformation_estimation.cu b/src/cupoch/registration/transformation_estimation.cu index daad683e..a99713a2 100644 --- a/src/cupoch/registration/transformation_estimation.cu +++ b/src/cupoch/registration/transformation_estimation.cu @@ -55,6 +55,52 @@ struct pt2pl_jacobian_residual_functor } }; +struct pl2pl_jacobian_residual_functor + : public utility::jacobian_residual_functor { + pl2pl_jacobian_residual_functor(const Eigen::Vector3f *source, + const Eigen::Vector3f *source_normals, + const Eigen::Vector3f *target_points, + const Eigen::Vector3f *target_normals, + const Eigen::Vector2i *corres) + : source_(source), + source_normals_(source_normals), + target_points_(target_points), + target_normals_(target_normals), + corres_(corres){}; + const Eigen::Vector3f *source_; + const Eigen::Vector3f *source_normals_; + const Eigen::Vector3f *target_points_; + const Eigen::Vector3f *target_normals_; + const Eigen::Vector2i *corres_; + __device__ void operator()(int idx, Eigen::Vector6f &vec, float &r) const { + const Eigen::Vector3f &vs = source_[corres_[idx][0]]; + const Eigen::Vector3f &ns = source_normals_[corres_[idx][0]]; + const Eigen::Vector3f &vt = target_points_[corres_[idx][1]]; + const Eigen::Vector3f &nt = target_normals_[corres_[idx][1]]; + + Eigen::Vector3f n = ns + nt; // sum of normals + Eigen::Vector3f d = vs - vt; // difference of points + + r = d.dot(n); // symmetric residual + + Eigen::Vector3f cross_product = (vs + vt).cross(n); + vec.block<3, 1>(0, 0) = cross_product; + vec.block<3, 1>(3, 0) = n; + } +}; + +// Define a function or a lambda to compute the error using normals +__device__ float ComputeErrorUsingNormals( + const Eigen::Vector3f &source_point, + const Eigen::Vector3f &source_normal, + const Eigen::Vector3f &target_point, + const Eigen::Vector3f &target_normal) { + // Implement the error calculation logic here + // This is just a placeholder logic + return (source_point - target_point).dot(target_normal) + + (source_point - target_point).dot(source_normal); +} + } // namespace float TransformationEstimationPointToPoint::ComputeRMSE( @@ -114,7 +160,8 @@ float TransformationEstimationPointToPlane::ComputeRMSE( target.normals_.begin(), thrust::make_transform_iterator( corres.begin(), - element_get_functor()))), + element_get_functor()))), make_tuple_iterator( thrust::make_permutation_iterator( source.points_.begin(), @@ -130,7 +177,8 @@ float TransformationEstimationPointToPlane::ComputeRMSE( target.normals_.begin(), thrust::make_transform_iterator( corres.end(), - element_get_functor()))), + element_get_functor()))), [] __device__(const thrust::tuple &x) { float r = (thrust::get<0>(x) - thrust::get<1>(x)) @@ -167,5 +215,100 @@ Eigen::Matrix4f TransformationEstimationPointToPlane::ComputeTransformation( utility::SolveJacobianSystemAndObtainExtrinsicMatrix(JTJ, JTr, det_thresh_); + return is_success ? extrinsic : Eigen::Matrix4f::Identity(); +} + +float TransformationEstimationPlaneToPlane::ComputeRMSE( + const geometry::PointCloud &source, + const geometry::PointCloud &target, + const CorrespondenceSet &corres) const { + if (corres.empty() || !source.HasNormals() || !target.HasNormals()) + return 0.0; + const float err = thrust::transform_reduce( + utility::exec_policy(0), + make_tuple_iterator( + thrust::make_permutation_iterator( + source.points_.begin(), + thrust::make_transform_iterator( + corres.begin(), + element_get_functor())), + thrust::make_permutation_iterator( + source.normals_.begin(), + thrust::make_transform_iterator( + corres.begin(), + element_get_functor())), + thrust::make_permutation_iterator( + target.points_.begin(), + thrust::make_transform_iterator( + corres.begin(), + element_get_functor())), + thrust::make_permutation_iterator( + target.normals_.begin(), + thrust::make_transform_iterator( + corres.begin(), + element_get_functor()))), + make_tuple_iterator( + thrust::make_permutation_iterator( + source.points_.begin(), + thrust::make_transform_iterator( + corres.end(), + element_get_functor())), + thrust::make_permutation_iterator( + source.normals_.begin(), + thrust::make_transform_iterator( + corres.end(), + element_get_functor())), + thrust::make_permutation_iterator( + target.points_.begin(), + thrust::make_transform_iterator( + corres.end(), + element_get_functor())), + thrust::make_permutation_iterator( + target.normals_.begin(), + thrust::make_transform_iterator( + corres.end(), + element_get_functor()))), + [] __device__( + const thrust::tuple &x) { + // Compute error using both source and target normals + float r = ComputeErrorUsingNormals( + thrust::get<0>(x), thrust::get<1>(x), thrust::get<2>(x), + thrust::get<3>(x)); + return r * r; + }, + 0.0f, thrust::plus()); + return std::sqrt(err / (float)corres.size()); +} + +Eigen::Matrix4f TransformationEstimationPlaneToPlane::ComputeTransformation( + const geometry::PointCloud &source, + const geometry::PointCloud &target, + const CorrespondenceSet &corres) const { + if (corres.empty() || !target.HasNormals()) + return Eigen::Matrix4f::Identity(); + + Eigen::Matrix6f JTJ; + Eigen::Vector6f JTr; + float r2; + pl2pl_jacobian_residual_functor func( + thrust::raw_pointer_cast(source.points_.data()), + thrust::raw_pointer_cast(source.normals_.data()), + thrust::raw_pointer_cast(target.points_.data()), + thrust::raw_pointer_cast(target.normals_.data()), + thrust::raw_pointer_cast(corres.data())); + thrust::tie(JTJ, JTr, r2) = + utility::ComputeJTJandJTr( + func, (int)corres.size()); + + bool is_success; + Eigen::Matrix4f extrinsic; + thrust::tie(is_success, extrinsic) = + utility::SolveJacobianSystemAndObtainExtrinsicMatrix(JTJ, JTr, + det_thresh_); + return is_success ? extrinsic : Eigen::Matrix4f::Identity(); } \ No newline at end of file diff --git a/src/cupoch/registration/transformation_estimation.h b/src/cupoch/registration/transformation_estimation.h index a0aa262b..84949248 100644 --- a/src/cupoch/registration/transformation_estimation.h +++ b/src/cupoch/registration/transformation_estimation.h @@ -39,7 +39,8 @@ enum class TransformationEstimationType { Unspecified = 0, PointToPoint = 1, PointToPlane = 2, - ColoredICP = 3, + PlaneToPlane = 3, + ColoredICP = 4, }; /// Base class that estimates a transformation between two point clouds @@ -113,5 +114,32 @@ class TransformationEstimationPointToPlane : public TransformationEstimation { TransformationEstimationType::PointToPlane; }; +/// Estimate a transformation for plane to plane distance +class TransformationEstimationPlaneToPlane : public TransformationEstimation { +public: + TransformationEstimationPlaneToPlane(float det_thresh = 1.0e-6) + : det_thresh_(det_thresh) {} + ~TransformationEstimationPlaneToPlane() override {} + +public: + TransformationEstimationType GetTransformationEstimationType() + const override { + return type_; + }; + float ComputeRMSE(const geometry::PointCloud &source, + const geometry::PointCloud &target, + const CorrespondenceSet &corres) const override; + Eigen::Matrix4f ComputeTransformation( + const geometry::PointCloud &source, + const geometry::PointCloud &target, + const CorrespondenceSet &corres) const override; + + float det_thresh_; + +private: + const TransformationEstimationType type_ = + TransformationEstimationType::PlaneToPlane; +}; + } // namespace registration } // namespace cupoch \ No newline at end of file diff --git a/src/python/cupoch_pybind/registration/registration.cpp b/src/python/cupoch_pybind/registration/registration.cpp index 2990b2b2..853eb31c 100644 --- a/src/python/cupoch_pybind/registration/registration.cpp +++ b/src/python/cupoch_pybind/registration/registration.cpp @@ -119,9 +119,10 @@ void pybind_registration_classes(py::module &m) { py::enum_ tf_type(m, "TransformationEstimationType"); tf_type.value("PointToPoint", registration::TransformationEstimationType::PointToPoint) - .value("PointToPlane", registration::TransformationEstimationType::PointToPlane) - .value("ColoredICP", registration::TransformationEstimationType::ColoredICP) - .export_values(); + .value("PointToPlane", registration::TransformationEstimationType::PointToPlane) + .value("PlaneToPlane", registration::TransformationEstimationType::PlaneToPlane) + .value("ColoredICP", registration::TransformationEstimationType::ColoredICP) + .export_values(); // cupoch.registration.TransformationEstimationPointToPoint: // TransformationEstimation @@ -164,6 +165,25 @@ void pybind_registration_classes(py::module &m) { return std::string("TransformationEstimationPointToPlane"); }); + // cupoch.registration.TransformationEstimationPlaneToPlane: + // TransformationEstimation + py::class_, + registration::TransformationEstimation> + te_l2l(m, "TransformationEstimationPlaneToPlane", + "Class to estimate a transformation for plane to plane " + "distance."); + py::detail::bind_default_constructor< + registration::TransformationEstimationPlaneToPlane>(te_l2l); + py::detail::bind_copy_functions< + registration::TransformationEstimationPlaneToPlane>(te_l2l); + te_l2l.def( + "__repr__", + [](const registration::TransformationEstimationPlaneToPlane &te) { + return std::string("TransformationEstimationPlaneToPlane"); + }); + // cupoch.registration.FastGlobalRegistrationOption: py::class_ fgr_option( m, "FastGlobalRegistrationOption", @@ -351,8 +371,9 @@ static const std::unordered_map {"criteria", "Convergence criteria"}, {"estimation_method", "Estimation method. One of " - "(``registration::TransformationEstimationPointToPoint``, " - "``registration::TransformationEstimationPointToPlane``)"}, + "(``registration::TransformationEstimationPointToPoint``," + "``registration::TransformationEstimationPointToPlane``," + "``registration::TransformationEstimationPlaneToPlane``)"}, {"init", "Initial transformation estimation"}, {"lambda_geometric", "lambda_geometric value"}, {"max_correspondence_distance", From f85a6a31b6f65696ed5e42f9009d9f5e6ef6c04c Mon Sep 17 00:00:00 2001 From: "sewon.jeon" Date: Fri, 8 Dec 2023 22:44:57 +0900 Subject: [PATCH 2/2] Rename to SymmetricMethod --- .../registration/transformation_estimation.cu | 4 ++-- .../registration/transformation_estimation.h | 10 +++++----- .../registration/registration.cpp | 20 +++++++++---------- 3 files changed, 17 insertions(+), 17 deletions(-) diff --git a/src/cupoch/registration/transformation_estimation.cu b/src/cupoch/registration/transformation_estimation.cu index a99713a2..a50934c3 100644 --- a/src/cupoch/registration/transformation_estimation.cu +++ b/src/cupoch/registration/transformation_estimation.cu @@ -218,7 +218,7 @@ Eigen::Matrix4f TransformationEstimationPointToPlane::ComputeTransformation( return is_success ? extrinsic : Eigen::Matrix4f::Identity(); } -float TransformationEstimationPlaneToPlane::ComputeRMSE( +float TransformationEstimationSymmetricMethod::ComputeRMSE( const geometry::PointCloud &source, const geometry::PointCloud &target, const CorrespondenceSet &corres) const { @@ -283,7 +283,7 @@ float TransformationEstimationPlaneToPlane::ComputeRMSE( return std::sqrt(err / (float)corres.size()); } -Eigen::Matrix4f TransformationEstimationPlaneToPlane::ComputeTransformation( +Eigen::Matrix4f TransformationEstimationSymmetricMethod::ComputeTransformation( const geometry::PointCloud &source, const geometry::PointCloud &target, const CorrespondenceSet &corres) const { diff --git a/src/cupoch/registration/transformation_estimation.h b/src/cupoch/registration/transformation_estimation.h index 84949248..fe344abd 100644 --- a/src/cupoch/registration/transformation_estimation.h +++ b/src/cupoch/registration/transformation_estimation.h @@ -39,7 +39,7 @@ enum class TransformationEstimationType { Unspecified = 0, PointToPoint = 1, PointToPlane = 2, - PlaneToPlane = 3, + SymmetricMethod = 3, ColoredICP = 4, }; @@ -115,11 +115,11 @@ class TransformationEstimationPointToPlane : public TransformationEstimation { }; /// Estimate a transformation for plane to plane distance -class TransformationEstimationPlaneToPlane : public TransformationEstimation { +class TransformationEstimationSymmetricMethod : public TransformationEstimation { public: - TransformationEstimationPlaneToPlane(float det_thresh = 1.0e-6) + TransformationEstimationSymmetricMethod(float det_thresh = 1.0e-6) : det_thresh_(det_thresh) {} - ~TransformationEstimationPlaneToPlane() override {} + ~TransformationEstimationSymmetricMethod() override {} public: TransformationEstimationType GetTransformationEstimationType() @@ -138,7 +138,7 @@ class TransformationEstimationPlaneToPlane : public TransformationEstimation { private: const TransformationEstimationType type_ = - TransformationEstimationType::PlaneToPlane; + TransformationEstimationType::SymmetricMethod; }; } // namespace registration diff --git a/src/python/cupoch_pybind/registration/registration.cpp b/src/python/cupoch_pybind/registration/registration.cpp index 853eb31c..f00ac38d 100644 --- a/src/python/cupoch_pybind/registration/registration.cpp +++ b/src/python/cupoch_pybind/registration/registration.cpp @@ -120,7 +120,7 @@ void pybind_registration_classes(py::module &m) { py::enum_ tf_type(m, "TransformationEstimationType"); tf_type.value("PointToPoint", registration::TransformationEstimationType::PointToPoint) .value("PointToPlane", registration::TransformationEstimationType::PointToPlane) - .value("PlaneToPlane", registration::TransformationEstimationType::PlaneToPlane) + .value("SymmetricMethod", registration::TransformationEstimationType::SymmetricMethod) .value("ColoredICP", registration::TransformationEstimationType::ColoredICP) .export_values(); @@ -165,23 +165,23 @@ void pybind_registration_classes(py::module &m) { return std::string("TransformationEstimationPointToPlane"); }); - // cupoch.registration.TransformationEstimationPlaneToPlane: + // cupoch.registration.TransformationEstimationSymmetricMethod: // TransformationEstimation - py::class_, + registration::TransformationEstimationSymmetricMethod>, registration::TransformationEstimation> - te_l2l(m, "TransformationEstimationPlaneToPlane", + te_l2l(m, "TransformationEstimationSymmetricMethod", "Class to estimate a transformation for plane to plane " "distance."); py::detail::bind_default_constructor< - registration::TransformationEstimationPlaneToPlane>(te_l2l); + registration::TransformationEstimationSymmetricMethod>(te_l2l); py::detail::bind_copy_functions< - registration::TransformationEstimationPlaneToPlane>(te_l2l); + registration::TransformationEstimationSymmetricMethod>(te_l2l); te_l2l.def( "__repr__", - [](const registration::TransformationEstimationPlaneToPlane &te) { - return std::string("TransformationEstimationPlaneToPlane"); + [](const registration::TransformationEstimationSymmetricMethod &te) { + return std::string("TransformationEstimationSymmetricMethod"); }); // cupoch.registration.FastGlobalRegistrationOption: @@ -373,7 +373,7 @@ static const std::unordered_map "Estimation method. One of " "(``registration::TransformationEstimationPointToPoint``," "``registration::TransformationEstimationPointToPlane``," - "``registration::TransformationEstimationPlaneToPlane``)"}, + "``registration::TransformationEstimationSymmetricMethod``)"}, {"init", "Initial transformation estimation"}, {"lambda_geometric", "lambda_geometric value"}, {"max_correspondence_distance",