389 lines
16 KiB
C++
389 lines
16 KiB
C++
/*
|
|
* Copyright 2016 The Cartographer Authors
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include "cartographer/mapping/internal/constraints/constraint_builder_3d.h"
|
|
|
|
#include <cmath>
|
|
#include <functional>
|
|
#include <iomanip>
|
|
#include <iostream>
|
|
#include <limits>
|
|
#include <memory>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <tuple>
|
|
#include <utility>
|
|
|
|
#include "Eigen/Eigenvalues"
|
|
#include "absl/memory/memory.h"
|
|
#include "cartographer/common/math.h"
|
|
#include "cartographer/common/thread_pool.h"
|
|
#include "cartographer/mapping/proto/scan_matching/ceres_scan_matcher_options_3d.pb.h"
|
|
#include "cartographer/mapping/proto/scan_matching/fast_correlative_scan_matcher_options_3d.pb.h"
|
|
#include "cartographer/metrics/counter.h"
|
|
#include "cartographer/metrics/gauge.h"
|
|
#include "cartographer/metrics/histogram.h"
|
|
#include "cartographer/transform/transform.h"
|
|
#include "glog/logging.h"
|
|
|
|
namespace cartographer {
|
|
namespace mapping {
|
|
namespace constraints {
|
|
|
|
static auto* kConstraintsSearchedMetric = metrics::Counter::Null();
|
|
static auto* kConstraintsFoundMetric = metrics::Counter::Null();
|
|
static auto* kGlobalConstraintsSearchedMetric = metrics::Counter::Null();
|
|
static auto* kGlobalConstraintsFoundMetric = metrics::Counter::Null();
|
|
static auto* kQueueLengthMetric = metrics::Gauge::Null();
|
|
static auto* kConstraintScoresMetric = metrics::Histogram::Null();
|
|
static auto* kConstraintRotationalScoresMetric = metrics::Histogram::Null();
|
|
static auto* kConstraintLowResolutionScoresMetric = metrics::Histogram::Null();
|
|
static auto* kGlobalConstraintScoresMetric = metrics::Histogram::Null();
|
|
static auto* kGlobalConstraintRotationalScoresMetric =
|
|
metrics::Histogram::Null();
|
|
static auto* kGlobalConstraintLowResolutionScoresMetric =
|
|
metrics::Histogram::Null();
|
|
static auto* kNumSubmapScanMatchersMetric = metrics::Gauge::Null();
|
|
|
|
ConstraintBuilder3D::ConstraintBuilder3D(
|
|
const proto::ConstraintBuilderOptions& options,
|
|
common::ThreadPoolInterface* const thread_pool)
|
|
: options_(options),
|
|
thread_pool_(thread_pool),
|
|
finish_node_task_(absl::make_unique<common::Task>()),
|
|
when_done_task_(absl::make_unique<common::Task>()),
|
|
ceres_scan_matcher_(options.ceres_scan_matcher_options_3d()) {}
|
|
|
|
ConstraintBuilder3D::~ConstraintBuilder3D() {
|
|
absl::MutexLock locker(&mutex_);
|
|
CHECK_EQ(finish_node_task_->GetState(), common::Task::NEW);
|
|
CHECK_EQ(when_done_task_->GetState(), common::Task::NEW);
|
|
CHECK_EQ(constraints_.size(), 0) << "WhenDone() was not called";
|
|
CHECK_EQ(num_started_nodes_, num_finished_nodes_);
|
|
CHECK(when_done_ == nullptr);
|
|
}
|
|
|
|
void ConstraintBuilder3D::MaybeAddConstraint(
|
|
const SubmapId& submap_id, const Submap3D* const submap,
|
|
const NodeId& node_id, const TrajectoryNode::Data* const constant_data,
|
|
const transform::Rigid3d& global_node_pose,
|
|
const transform::Rigid3d& global_submap_pose) {
|
|
if ((global_node_pose.translation() - global_submap_pose.translation())
|
|
.norm() > options_.max_constraint_distance()) {
|
|
return;
|
|
}
|
|
if (!per_submap_sampler_
|
|
.emplace(std::piecewise_construct, std::forward_as_tuple(submap_id),
|
|
std::forward_as_tuple(options_.sampling_ratio()))
|
|
.first->second.Pulse()) {
|
|
return;
|
|
}
|
|
|
|
absl::MutexLock locker(&mutex_);
|
|
if (when_done_) {
|
|
LOG(WARNING)
|
|
<< "MaybeAddConstraint was called while WhenDone was scheduled.";
|
|
}
|
|
constraints_.emplace_back();
|
|
kQueueLengthMetric->Set(constraints_.size());
|
|
auto* const constraint = &constraints_.back();
|
|
const auto* scan_matcher = DispatchScanMatcherConstruction(submap_id, submap);
|
|
auto constraint_task = absl::make_unique<common::Task>();
|
|
constraint_task->SetWorkItem([=]() LOCKS_EXCLUDED(mutex_) {
|
|
ComputeConstraint(submap_id, node_id, false, /* match_full_submap */
|
|
constant_data, global_node_pose, global_submap_pose,
|
|
*scan_matcher, constraint);
|
|
});
|
|
constraint_task->AddDependency(scan_matcher->creation_task_handle);
|
|
auto constraint_task_handle =
|
|
thread_pool_->Schedule(std::move(constraint_task));
|
|
finish_node_task_->AddDependency(constraint_task_handle);
|
|
}
|
|
|
|
void ConstraintBuilder3D::MaybeAddGlobalConstraint(
|
|
const SubmapId& submap_id, const Submap3D* const submap,
|
|
const NodeId& node_id, const TrajectoryNode::Data* const constant_data,
|
|
const Eigen::Quaterniond& global_node_rotation,
|
|
const Eigen::Quaterniond& global_submap_rotation) {
|
|
absl::MutexLock locker(&mutex_);
|
|
if (when_done_) {
|
|
LOG(WARNING)
|
|
<< "MaybeAddGlobalConstraint was called while WhenDone was scheduled.";
|
|
}
|
|
constraints_.emplace_back();
|
|
kQueueLengthMetric->Set(constraints_.size());
|
|
auto* const constraint = &constraints_.back();
|
|
const auto* scan_matcher = DispatchScanMatcherConstruction(submap_id, submap);
|
|
auto constraint_task = absl::make_unique<common::Task>();
|
|
constraint_task->SetWorkItem([=]() LOCKS_EXCLUDED(mutex_) {
|
|
ComputeConstraint(submap_id, node_id, true, /* match_full_submap */
|
|
constant_data,
|
|
transform::Rigid3d::Rotation(global_node_rotation),
|
|
transform::Rigid3d::Rotation(global_submap_rotation),
|
|
*scan_matcher, constraint);
|
|
});
|
|
constraint_task->AddDependency(scan_matcher->creation_task_handle);
|
|
auto constraint_task_handle =
|
|
thread_pool_->Schedule(std::move(constraint_task));
|
|
finish_node_task_->AddDependency(constraint_task_handle);
|
|
}
|
|
|
|
void ConstraintBuilder3D::NotifyEndOfNode() {
|
|
absl::MutexLock locker(&mutex_);
|
|
CHECK(finish_node_task_ != nullptr);
|
|
finish_node_task_->SetWorkItem([this] {
|
|
absl::MutexLock locker(&mutex_);
|
|
++num_finished_nodes_;
|
|
});
|
|
auto finish_node_task_handle =
|
|
thread_pool_->Schedule(std::move(finish_node_task_));
|
|
finish_node_task_ = absl::make_unique<common::Task>();
|
|
when_done_task_->AddDependency(finish_node_task_handle);
|
|
++num_started_nodes_;
|
|
}
|
|
|
|
void ConstraintBuilder3D::WhenDone(
|
|
const std::function<void(const ConstraintBuilder3D::Result&)>& callback) {
|
|
absl::MutexLock locker(&mutex_);
|
|
CHECK(when_done_ == nullptr);
|
|
// TODO(gaschler): Consider using just std::function, it can also be empty.
|
|
when_done_ = absl::make_unique<std::function<void(const Result&)>>(callback);
|
|
CHECK(when_done_task_ != nullptr);
|
|
when_done_task_->SetWorkItem([this] { RunWhenDoneCallback(); });
|
|
thread_pool_->Schedule(std::move(when_done_task_));
|
|
when_done_task_ = absl::make_unique<common::Task>();
|
|
}
|
|
|
|
const ConstraintBuilder3D::SubmapScanMatcher*
|
|
ConstraintBuilder3D::DispatchScanMatcherConstruction(const SubmapId& submap_id,
|
|
const Submap3D* submap) {
|
|
if (submap_scan_matchers_.count(submap_id) != 0) {
|
|
return &submap_scan_matchers_.at(submap_id);
|
|
}
|
|
auto& submap_scan_matcher = submap_scan_matchers_[submap_id];
|
|
kNumSubmapScanMatchersMetric->Set(submap_scan_matchers_.size());
|
|
submap_scan_matcher.high_resolution_hybrid_grid =
|
|
&submap->high_resolution_hybrid_grid();
|
|
submap_scan_matcher.low_resolution_hybrid_grid =
|
|
&submap->low_resolution_hybrid_grid();
|
|
auto& scan_matcher_options =
|
|
options_.fast_correlative_scan_matcher_options_3d();
|
|
const Eigen::VectorXf* histogram =
|
|
&submap->rotational_scan_matcher_histogram();
|
|
auto scan_matcher_task = absl::make_unique<common::Task>();
|
|
scan_matcher_task->SetWorkItem(
|
|
[&submap_scan_matcher, &scan_matcher_options, histogram]() {
|
|
submap_scan_matcher.fast_correlative_scan_matcher =
|
|
absl::make_unique<scan_matching::FastCorrelativeScanMatcher3D>(
|
|
*submap_scan_matcher.high_resolution_hybrid_grid,
|
|
submap_scan_matcher.low_resolution_hybrid_grid, histogram,
|
|
scan_matcher_options);
|
|
});
|
|
submap_scan_matcher.creation_task_handle =
|
|
thread_pool_->Schedule(std::move(scan_matcher_task));
|
|
return &submap_scan_matchers_.at(submap_id);
|
|
}
|
|
|
|
void ConstraintBuilder3D::ComputeConstraint(
|
|
const SubmapId& submap_id, const NodeId& node_id, bool match_full_submap,
|
|
const TrajectoryNode::Data* const constant_data,
|
|
const transform::Rigid3d& global_node_pose,
|
|
const transform::Rigid3d& global_submap_pose,
|
|
const SubmapScanMatcher& submap_scan_matcher,
|
|
std::unique_ptr<Constraint>* constraint) {
|
|
CHECK(submap_scan_matcher.fast_correlative_scan_matcher);
|
|
// The 'constraint_transform' (submap i <- node j) is computed from:
|
|
// - a 'high_resolution_point_cloud' in node j and
|
|
// - the initial guess 'initial_pose' (submap i <- node j).
|
|
std::unique_ptr<scan_matching::FastCorrelativeScanMatcher3D::Result>
|
|
match_result;
|
|
|
|
// Compute 'pose_estimate' in three stages:
|
|
// 1. Fast estimate using the fast correlative scan matcher.
|
|
// 2. Prune if the score is too low.
|
|
// 3. Refine.
|
|
if (match_full_submap) {
|
|
kGlobalConstraintsSearchedMetric->Increment();
|
|
match_result =
|
|
submap_scan_matcher.fast_correlative_scan_matcher->MatchFullSubmap(
|
|
global_node_pose.rotation(), global_submap_pose.rotation(),
|
|
*constant_data, options_.global_localization_min_score());
|
|
if (match_result != nullptr) {
|
|
CHECK_GT(match_result->score, options_.global_localization_min_score());
|
|
CHECK_GE(node_id.trajectory_id, 0);
|
|
CHECK_GE(submap_id.trajectory_id, 0);
|
|
kGlobalConstraintsFoundMetric->Increment();
|
|
kGlobalConstraintScoresMetric->Observe(match_result->score);
|
|
kGlobalConstraintRotationalScoresMetric->Observe(
|
|
match_result->rotational_score);
|
|
kGlobalConstraintLowResolutionScoresMetric->Observe(
|
|
match_result->low_resolution_score);
|
|
} else {
|
|
return;
|
|
}
|
|
} else {
|
|
kConstraintsSearchedMetric->Increment();
|
|
match_result = submap_scan_matcher.fast_correlative_scan_matcher->Match(
|
|
global_node_pose, global_submap_pose, *constant_data,
|
|
options_.min_score());
|
|
if (match_result != nullptr) {
|
|
// We've reported a successful local match.
|
|
CHECK_GT(match_result->score, options_.min_score());
|
|
kConstraintsFoundMetric->Increment();
|
|
kConstraintScoresMetric->Observe(match_result->score);
|
|
kConstraintRotationalScoresMetric->Observe(
|
|
match_result->rotational_score);
|
|
kConstraintLowResolutionScoresMetric->Observe(
|
|
match_result->low_resolution_score);
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
{
|
|
absl::MutexLock locker(&mutex_);
|
|
score_histogram_.Add(match_result->score);
|
|
rotational_score_histogram_.Add(match_result->rotational_score);
|
|
low_resolution_score_histogram_.Add(match_result->low_resolution_score);
|
|
}
|
|
|
|
// Use the CSM estimate as both the initial and previous pose. This has the
|
|
// effect that, in the absence of better information, we prefer the original
|
|
// CSM estimate.
|
|
ceres::Solver::Summary unused_summary;
|
|
transform::Rigid3d constraint_transform;
|
|
ceres_scan_matcher_.Match(match_result->pose_estimate.translation(),
|
|
match_result->pose_estimate,
|
|
{{&constant_data->high_resolution_point_cloud,
|
|
submap_scan_matcher.high_resolution_hybrid_grid},
|
|
{&constant_data->low_resolution_point_cloud,
|
|
submap_scan_matcher.low_resolution_hybrid_grid}},
|
|
&constraint_transform, &unused_summary);
|
|
|
|
constraint->reset(new Constraint{
|
|
submap_id,
|
|
node_id,
|
|
{constraint_transform, options_.loop_closure_translation_weight(),
|
|
options_.loop_closure_rotation_weight()},
|
|
Constraint::INTER_SUBMAP});
|
|
|
|
if (options_.log_matches()) {
|
|
std::ostringstream info;
|
|
info << "Node " << node_id << " with "
|
|
<< constant_data->high_resolution_point_cloud.size()
|
|
<< " points on submap " << submap_id << std::fixed;
|
|
if (match_full_submap) {
|
|
info << " matches";
|
|
} else {
|
|
// Compute the difference between (submap i <- node j) according to loop
|
|
// closure ('constraint_transform') and according to global SLAM state.
|
|
const transform::Rigid3d difference = global_node_pose.inverse() *
|
|
global_submap_pose *
|
|
constraint_transform;
|
|
info << " differs by translation " << std::setprecision(2)
|
|
<< difference.translation().norm() << " rotation "
|
|
<< std::setprecision(3) << transform::GetAngle(difference);
|
|
}
|
|
info << " with score " << std::setprecision(1) << 100. * match_result->score
|
|
<< "%.";
|
|
LOG(INFO) << info.str();
|
|
}
|
|
}
|
|
|
|
void ConstraintBuilder3D::RunWhenDoneCallback() {
|
|
Result result;
|
|
std::unique_ptr<std::function<void(const Result&)>> callback;
|
|
{
|
|
absl::MutexLock locker(&mutex_);
|
|
CHECK(when_done_ != nullptr);
|
|
for (const std::unique_ptr<Constraint>& constraint : constraints_) {
|
|
if (constraint == nullptr) continue;
|
|
result.push_back(*constraint);
|
|
}
|
|
if (options_.log_matches()) {
|
|
LOG(INFO) << constraints_.size() << " computations resulted in "
|
|
<< result.size() << " additional constraints.\n"
|
|
<< "Score histogram:\n"
|
|
<< score_histogram_.ToString(10) << "\n"
|
|
<< "Rotational score histogram:\n"
|
|
<< rotational_score_histogram_.ToString(10) << "\n"
|
|
<< "Low resolution score histogram:\n"
|
|
<< low_resolution_score_histogram_.ToString(10);
|
|
}
|
|
constraints_.clear();
|
|
callback = std::move(when_done_);
|
|
when_done_.reset();
|
|
kQueueLengthMetric->Set(constraints_.size());
|
|
}
|
|
(*callback)(result);
|
|
}
|
|
|
|
int ConstraintBuilder3D::GetNumFinishedNodes() {
|
|
absl::MutexLock locker(&mutex_);
|
|
return num_finished_nodes_;
|
|
}
|
|
|
|
void ConstraintBuilder3D::DeleteScanMatcher(const SubmapId& submap_id) {
|
|
absl::MutexLock locker(&mutex_);
|
|
if (when_done_) {
|
|
LOG(WARNING)
|
|
<< "DeleteScanMatcher was called while WhenDone was scheduled.";
|
|
}
|
|
submap_scan_matchers_.erase(submap_id);
|
|
per_submap_sampler_.erase(submap_id);
|
|
kNumSubmapScanMatchersMetric->Set(submap_scan_matchers_.size());
|
|
}
|
|
|
|
void ConstraintBuilder3D::RegisterMetrics(metrics::FamilyFactory* factory) {
|
|
auto* counts = factory->NewCounterFamily(
|
|
"mapping_constraints_constraint_builder_3d_constraints",
|
|
"Constraints computed");
|
|
kConstraintsSearchedMetric =
|
|
counts->Add({{"search_region", "local"}, {"matcher", "searched"}});
|
|
kConstraintsFoundMetric =
|
|
counts->Add({{"search_region", "local"}, {"matcher", "found"}});
|
|
kGlobalConstraintsSearchedMetric =
|
|
counts->Add({{"search_region", "global"}, {"matcher", "searched"}});
|
|
kGlobalConstraintsFoundMetric =
|
|
counts->Add({{"search_region", "global"}, {"matcher", "found"}});
|
|
auto* queue_length = factory->NewGaugeFamily(
|
|
"mapping_constraints_constraint_builder_3d_queue_length", "Queue length");
|
|
kQueueLengthMetric = queue_length->Add({});
|
|
auto boundaries = metrics::Histogram::FixedWidth(0.05, 20);
|
|
auto* scores = factory->NewHistogramFamily(
|
|
"mapping_constraints_constraint_builder_3d_scores",
|
|
"Constraint scores built", boundaries);
|
|
kConstraintScoresMetric =
|
|
scores->Add({{"search_region", "local"}, {"kind", "score"}});
|
|
kConstraintRotationalScoresMetric =
|
|
scores->Add({{"search_region", "local"}, {"kind", "rotational_score"}});
|
|
kConstraintLowResolutionScoresMetric = scores->Add(
|
|
{{"search_region", "local"}, {"kind", "low_resolution_score"}});
|
|
kGlobalConstraintScoresMetric =
|
|
scores->Add({{"search_region", "global"}, {"kind", "score"}});
|
|
kGlobalConstraintRotationalScoresMetric =
|
|
scores->Add({{"search_region", "global"}, {"kind", "rotational_score"}});
|
|
kGlobalConstraintLowResolutionScoresMetric = scores->Add(
|
|
{{"search_region", "global"}, {"kind", "low_resolution_score"}});
|
|
auto* num_matchers = factory->NewGaugeFamily(
|
|
"mapping_constraints_constraint_builder_3d_num_submap_scan_matchers",
|
|
"Current number of constructed submap scan matchers");
|
|
kNumSubmapScanMatchersMetric = num_matchers->Add({});
|
|
}
|
|
|
|
} // namespace constraints
|
|
} // namespace mapping
|
|
} // namespace cartographer
|