From 0ceec7d7da2fe387c0d4e1bf872d2184c79a7d1a Mon Sep 17 00:00:00 2001 From: feixyz10 Date: Tue, 26 Jan 2021 21:00:31 +0800 Subject: [PATCH] mapper: coding... --- CMakeLists.txt | 2 +- oh_my_loam/CMakeLists.txt | 2 +- oh_my_loam/extractor/extractor_VLP16.h | 2 +- oh_my_loam/mapper/mapper.cc | 15 +++++- oh_my_loam/mapper/mapper.h | 18 +++++--- oh_my_loam/odometer/odometer.cc | 20 +++----- oh_my_loam/odometer/odometer.h | 12 +++-- oh_my_loam/oh_my_loam.cc | 48 +++++++++++++++----- oh_my_loam/oh_my_loam.h | 21 +++++++-- oh_my_loam/visualizer/ohmyloam_visualizer.cc | 39 ++++++++++++++++ oh_my_loam/visualizer/ohmyloam_visualizer.h | 29 ++++++++++++ 11 files changed, 164 insertions(+), 44 deletions(-) create mode 100644 oh_my_loam/visualizer/ohmyloam_visualizer.cc create mode 100644 oh_my_loam/visualizer/ohmyloam_visualizer.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 974744d..f71d4ae 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -49,7 +49,7 @@ target_link_libraries(main oh_my_loam extractor odometer - # mapper + mapper solver ${CERES_LIBRARIES} visualizer diff --git a/oh_my_loam/CMakeLists.txt b/oh_my_loam/CMakeLists.txt index a741e62..56730c2 100644 --- a/oh_my_loam/CMakeLists.txt +++ b/oh_my_loam/CMakeLists.txt @@ -2,7 +2,7 @@ add_subdirectory(base) add_subdirectory(extractor) add_subdirectory(visualizer) add_subdirectory(odometer) -# add_subdirectory(mapper) +add_subdirectory(mapper) add_subdirectory(solver) aux_source_directory(. SRC_FILES) diff --git a/oh_my_loam/extractor/extractor_VLP16.h b/oh_my_loam/extractor/extractor_VLP16.h index de5b44f..6986261 100644 --- a/oh_my_loam/extractor/extractor_VLP16.h +++ b/oh_my_loam/extractor/extractor_VLP16.h @@ -1,6 +1,6 @@ #pragma once -#include "extractor.h" +#include "oh_my_loam/extractor/extractor.h" namespace oh_my_loam { diff --git a/oh_my_loam/mapper/mapper.cc b/oh_my_loam/mapper/mapper.cc index 4fab450..8baa37f 100644 --- a/oh_my_loam/mapper/mapper.cc +++ b/oh_my_loam/mapper/mapper.cc @@ -15,8 +15,19 @@ bool Mapper::Init() { return true; } -void Mapper::Process(double timestamp, const std::vector &features, - Pose3d *const pose_out) {} +void Mapper::Reset() {} + +void Mapper::Process(double timestamp, const TPointCloudConstPtr &cloud_corn, + const TPointCloudConstPtr &cloud_surf, + common::Pose3d *const pose_out) { + if (!is_initialized_) { + cloud_corn_map_ = cloud_corn; + cloud_surf_map_ = cloud_surf; + pose_out->SetIdentity(); + is_initialized_ = true; + return; + } +} void Mapper::Visualize() {} diff --git a/oh_my_loam/mapper/mapper.h b/oh_my_loam/mapper/mapper.h index 3261002..c7e40b7 100644 --- a/oh_my_loam/mapper/mapper.h +++ b/oh_my_loam/mapper/mapper.h @@ -15,11 +15,16 @@ class Mapper { bool Init(); - void Process(double timestamp, const std::vector &features, + void Process(double timestamp, const TPointCloudConstPtr &cloud_corn, + const TPointCloudConstPtr &cloud_surf, common::Pose3d *const pose_out); - common::PointCloudConstPtr map() const { - return cloud_map_; + TPointCloudConstPtr cloud_corn_map() const { + return cloud_corn_map_; + } + + TPointCloudConstPtr cloud_surf_map() const { + return cloud_corn_map_; } void Reset(); @@ -27,13 +32,12 @@ class Mapper { private: void Visualize(); - common::PointCloudPtr cloud_map_; - - common::Pose3d pose_curr2world_; + TPointCloudPtr cloud_corn_map_; + TPointCloudPtr cloud_surf_map_; YAML::Node config_; - bool is_initialized = false; + bool is_initialized_ = false; bool is_vis_ = false; diff --git a/oh_my_loam/odometer/odometer.cc b/oh_my_loam/odometer/odometer.cc index 40798c8..e903077 100644 --- a/oh_my_loam/odometer/odometer.cc +++ b/oh_my_loam/odometer/odometer.cc @@ -36,10 +36,6 @@ void Odometer::Process(double timestamp, const std::vector &features, return; } BLOCK_TIMER_START; - const auto &cloud_corn = kdtree_corn_.getInputCloud(); - const auto &cloud_surf = kdtree_surf_.getInputCloud(); - AINFO_IF(verbose_) << "Points to be matched: " << cloud_corn->size() << " + " - << cloud_surf->size(); std::vector pp_pairs; std::vector pl_pairs; for (int i = 0; i < config_["icp_iter_num"].as(); ++i) { @@ -71,13 +67,13 @@ void Odometer::Process(double timestamp, const std::vector &features, AINFO_IF(verbose_) << "Odometer::ICP: iter_" << i << ": " << BLOCK_TIMER_STOP_FMT; } - pose_curr2world_ = pose_curr2world_ * pose_curr2last_; - *pose_out = pose_curr2world_; + *pose_out = *pose_out * pose_curr2last_; + pose_curr2world_ = *pose_out; AINFO_IF(verbose_) << "Pose increase: " << pose_curr2last_.ToString(); - AINFO_IF(verbose_) << "Pose after: " << pose_curr2world_.ToString(); + // mush called before calling UpdatePre + if (is_vis_) Visualize(pl_pairs, pp_pairs); UpdatePre(features); AINFO << "Odometer::Porcess: " << BLOCK_TIMER_STOP_FMT; - if (is_vis_) Visualize(cloud_corn, cloud_surf, pl_pairs, pp_pairs); } void Odometer::MatchCorn(const TPointCloud &src, @@ -191,15 +187,13 @@ void Odometer::UpdatePre(const std::vector &features) { kdtree_surf_.setInputCloud(surf_pre); } -void Odometer::Visualize(const TPointCloudConstPtr &cloud_corn, - const TPointCloudConstPtr &cloud_surf, - const std::vector &pl_pairs, +void Odometer::Visualize(const std::vector &pl_pairs, const std::vector &pp_pairs, double timestamp) const { std::shared_ptr frame(new OdometerVisFrame); frame->timestamp = timestamp; - frame->cloud_corn = cloud_corn; - frame->cloud_surf = cloud_surf; + frame->cloud_corn = kdtree_corn_.getInputCloud(); + frame->cloud_surf = kdtree_surf_.getInputCloud(); frame->pl_pairs = pl_pairs; frame->pp_pairs = pp_pairs; frame->pose_curr2last = pose_curr2last_; diff --git a/oh_my_loam/odometer/odometer.h b/oh_my_loam/odometer/odometer.h index 8c02769..83cd8c6 100644 --- a/oh_my_loam/odometer/odometer.h +++ b/oh_my_loam/odometer/odometer.h @@ -18,6 +18,14 @@ class Odometer { void Process(double timestamp, const std::vector &features, common::Pose3d *const pose_out); + TPointCloudConstPtr cloud_corn() const { + return kdtree_corn_.getInputCloud(); + } + + TPointCloudConstPtr cloud_surf() const { + return kdtree_surf_.getInputCloud(); + } + void Reset(); protected: @@ -29,9 +37,7 @@ class Odometer { void MatchSurf(const TPointCloud &src, std::vector *const pairs) const; - void Visualize(const TPointCloudConstPtr &cloud_corn, - const TPointCloudConstPtr &cloud_surf, - const std::vector &pl_pairs, + void Visualize(const std::vector &pl_pairs, const std::vector &pp_pairs, double timestamp = 0.0) const; diff --git a/oh_my_loam/oh_my_loam.cc b/oh_my_loam/oh_my_loam.cc index 73b4503..be83011 100644 --- a/oh_my_loam/oh_my_loam.cc +++ b/oh_my_loam/oh_my_loam.cc @@ -33,11 +33,12 @@ bool OhMyLoam::Init() { } void OhMyLoam::Reset() { - timestamp_last_ = timestamp_last_mapping_ = 0.0; extractor_->Reset(); odometer_->Reset(); mapper_->Reset(); - std::vector().swap(poses_); + std::vector().swap(poses_curr2world_); + std::lock_guard lock(mutex_); + pose_mapping_updated_ = true; } void OhMyLoam::Run(double timestamp, @@ -46,13 +47,41 @@ void OhMyLoam::Run(double timestamp, RemoveOutliers(*cloud_in, cloud.get()); std::vector features; extractor_->Process(timestamp, cloud, &features); - Pose3d pose; - odometer_->Process(timestamp, features, &pose); - poses_.emplace_back(pose); - if (!IsMapping(timestamp)) return; - mapper_->Process(); + FusionOdometryMapping(); + auto pose_odom = + poses_curr2world_.empty() ? TimePose() : poses_curr2world_.back(); + odometer_->Process(timestamp, features, &pose_odom.pose); + poses_curr2world_.push_back(pose_odom); + const auto &cloud_corn = odometer_->cloud_corn(); + const auto &cloud_surf = odometer_->cloud_surf(); + + if (!pose_mapping_updated_) return; + mapping_thread_.reset(new std::thread(&OhMyLoam::MappingProcess, this, + timestamp, cloud_corn, cloud_surf)); + if (mapping_thread_->joinable()) mapping_thread_->detach(); } +void OhMyLoam::MappingProcess(double timestamp, + const TPointCloudConstPtr &cloud_corn, + const TPointCloudConstPtr &cloud_surf) { + TimePose pose; + pose.timestamp = timestamp; + mapper_->Process(timestamp, cloud_corn, cloud_corn, &pose.pose); + std::lock_guard lock(mutex_); + pose_mapping_ = pose; + pose_mapping_updated_ = true; +} + +void OhMyLoam::FusionOdometryMapping() { + std::lock_guard lock(mutex_); + TimePose pose_m = pose_mapping_; + pose_mapping_updated_ = false; + for (;;) { + } +} + +void OhMyLoam::Visualize(double timestamp) {} + void OhMyLoam::RemoveOutliers(const common::PointCloud &cloud_in, common::PointCloud *const cloud_out) const { common::RemovePoints(cloud_in, cloud_out, [&](const auto &pt) { @@ -62,9 +91,4 @@ void OhMyLoam::RemoveOutliers(const common::PointCloud &cloud_in, }); } -bool OhMyLoam::IsMapping(double timestamp) const { - return std::abs(timestamp - timestamp_last_mapping_) >= - config_["mapper_config"]["process_period"].as(); -} - } // namespace oh_my_loam \ No newline at end of file diff --git a/oh_my_loam/oh_my_loam.h b/oh_my_loam/oh_my_loam.h index 56001d0..38ee88c 100644 --- a/oh_my_loam/oh_my_loam.h +++ b/oh_my_loam/oh_my_loam.h @@ -20,7 +20,12 @@ class OhMyLoam { private: void Reset(); - bool IsMapping(double timestamp) const; + void FusionOdometryMapping(); + + void MappingProcess(double timestamp, const TPointCloudConstPtr &cloud_corn, + const TPointCloudConstPtr &cloud_surf); + + void Visualize(double timestamp = 0.0); std::unique_ptr extractor_{nullptr}; @@ -32,12 +37,20 @@ class OhMyLoam { void RemoveOutliers(const common::PointCloud &cloud_in, common::PointCloud *const cloud_out) const; - std::vector poses_; + struct TimePose { + double timestamp; + common::Pose3d pose; + }; + + std::vector poses_curr2world_; + + std::mutex mutex_; + TimePose pose_mapping_; + bool pose_mapping_updated_ = true; + std::unique_ptr mapping_thread_{nullptr}; YAML::Node config_; - double timestamp_last_ = 0.0, timestamp_last_mapping_ = 0.0; - DISALLOW_COPY_AND_ASSIGN(OhMyLoam) }; diff --git a/oh_my_loam/visualizer/ohmyloam_visualizer.cc b/oh_my_loam/visualizer/ohmyloam_visualizer.cc new file mode 100644 index 0000000..12a61d4 --- /dev/null +++ b/oh_my_loam/visualizer/ohmyloam_visualizer.cc @@ -0,0 +1,39 @@ +#include "ohmyloam_visualizer.h" + +#include "common/color/color.h" + +namespace oh_my_loam { + +namespace { +using namespace common; +#define LIGHT_BLUE common::Color(0, 0, 80) +#define LIGHT_YELLOW common::Color(80, 80, 0) +} // namespace + +void OhmyloamVisualizer::Draw() { + auto frame = GetCurrentFrame(); + DrawPointCloud(frame.cloud_map_corn, LIGHT_YELLOW, "cloud_map_corn"); + DrawPointCloud(frame.cloud_map_surf, LIGHT_BLUE, "cloud_map_surf"); + + DrawPointCloud(frame.cloud_corn, RED, "cloud_corn"); + DrawPointCloud(frame.cloud_surf, CYAN, "cloud_surf"); + + DrawTrajectory(frame.poses); +} + +void OhmyloamVisualizer::DrawTrajectory(std::vector &poses) { + std::vector poses_n; + poses_n.reserve((poses.size())); + Pose3d pose_inv = poses.back().Inv(); + for (const auto &pose : poses) { + poses_n.emplace_back(pose_inv * pose); + }; + for (size_t i = 0; i < poses_n.size() - 1; ++i) { + Eigen::Vector3f p1 = poses_n[i].t_vec().cast(); + Eigen::Vector3f p2 = poses_n[i + 1].t_vec().cast(); + AddLine({p1.x(), p1.y(), p1.z()}, {p2.x(), p2.y(), p2.z()}, PINK, + "trajectory" + std::to_string(i), viewer_.get()); + } +} + +} // namespace oh_my_loam \ No newline at end of file diff --git a/oh_my_loam/visualizer/ohmyloam_visualizer.h b/oh_my_loam/visualizer/ohmyloam_visualizer.h new file mode 100644 index 0000000..34cc0f3 --- /dev/null +++ b/oh_my_loam/visualizer/ohmyloam_visualizer.h @@ -0,0 +1,29 @@ +#pragma once + +#include "common/geometry/pose3d.h" +#include "common/visualizer/lidar_visualizer.h" +#include "oh_my_loam/base/types.h" + +namespace oh_my_loam { + +struct OhmyloamVisFrame : public common::LidarVisFrame { + TPointCloudConstPtr cloud_map_corn; + TPointCloudConstPtr cloud_map_surf; + TPointCloudConstPtr cloud_corn; // current + TPointCloudConstPtr cloud_surf; // current + std::vector poses; +}; + +class OhmyloamVisualizer : public common::LidarVisualizer { + public: + explicit OhmyloamVisualizer(const std::string &name = "OhmyloamVisualizer", + size_t max_history_size = 10) + : common::LidarVisualizer(name, max_history_size) {} + + private: + void Draw() override; + + void DrawTrajectory(std::vector &poses); +}; + +} // namespace oh_my_loam