243 lines
10 KiB
C++
243 lines
10 KiB
C++
/*
|
|
* Copyright 2018 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/2d/tsdf_range_data_inserter_2d.h"
|
|
|
|
#include "cartographer/mapping/internal/2d/normal_estimation_2d.h"
|
|
#include "cartographer/mapping/internal/2d/ray_to_pixel_mask.h"
|
|
|
|
namespace cartographer {
|
|
namespace mapping {
|
|
namespace {
|
|
|
|
// Factor for subpixel accuracy of start and end point for ray casts.
|
|
constexpr int kSubpixelScale = 1000;
|
|
// Minimum distance between range observation and origin. Otherwise, range
|
|
// observations are discarded.
|
|
constexpr float kMinRangeMeters = 1e-6f;
|
|
const float kSqrtTwoPi = std::sqrt(2.0 * M_PI);
|
|
|
|
void GrowAsNeeded(const sensor::RangeData& range_data,
|
|
const float truncation_distance, TSDF2D* const tsdf) {
|
|
Eigen::AlignedBox2f bounding_box(range_data.origin.head<2>());
|
|
for (const sensor::RangefinderPoint& hit : range_data.returns) {
|
|
const Eigen::Vector3f direction =
|
|
(hit.position - range_data.origin).normalized();
|
|
const Eigen::Vector3f end_position =
|
|
hit.position + truncation_distance * direction;
|
|
bounding_box.extend(end_position.head<2>());
|
|
}
|
|
// Padding around bounding box to avoid numerical issues at cell boundaries.
|
|
constexpr float kPadding = 1e-6f;
|
|
tsdf->GrowLimits(bounding_box.min() - kPadding * Eigen::Vector2f::Ones());
|
|
tsdf->GrowLimits(bounding_box.max() + kPadding * Eigen::Vector2f::Ones());
|
|
}
|
|
|
|
float GaussianKernel(const float x, const float sigma) {
|
|
return 1.0 / (kSqrtTwoPi * sigma) * std::exp(-0.5 * x * x / (sigma * sigma));
|
|
}
|
|
|
|
std::pair<Eigen::Array2i, Eigen::Array2i> SuperscaleRay(
|
|
const Eigen::Vector2f& begin, const Eigen::Vector2f& end,
|
|
TSDF2D* const tsdf) {
|
|
const MapLimits& limits = tsdf->limits();
|
|
const double superscaled_resolution = limits.resolution() / kSubpixelScale;
|
|
const MapLimits superscaled_limits(
|
|
superscaled_resolution, limits.max(),
|
|
CellLimits(limits.cell_limits().num_x_cells * kSubpixelScale,
|
|
limits.cell_limits().num_y_cells * kSubpixelScale));
|
|
|
|
const Eigen::Array2i superscaled_begin =
|
|
superscaled_limits.GetCellIndex(begin);
|
|
const Eigen::Array2i superscaled_end = superscaled_limits.GetCellIndex(end);
|
|
return std::make_pair(superscaled_begin, superscaled_end);
|
|
}
|
|
|
|
struct RangeDataSorter {
|
|
RangeDataSorter(Eigen::Vector3f origin) { origin_ = origin.head<2>(); }
|
|
bool operator()(const sensor::RangefinderPoint& lhs,
|
|
const sensor::RangefinderPoint& rhs) {
|
|
const Eigen::Vector2f delta_lhs =
|
|
(lhs.position.head<2>() - origin_).normalized();
|
|
const Eigen::Vector2f delta_rhs =
|
|
(rhs.position.head<2>() - origin_).normalized();
|
|
if ((delta_lhs[1] < 0.f) != (delta_rhs[1] < 0.f)) {
|
|
return delta_lhs[1] < 0.f;
|
|
} else if (delta_lhs[1] < 0.f) {
|
|
return delta_lhs[0] < delta_rhs[0];
|
|
} else {
|
|
return delta_lhs[0] > delta_rhs[0];
|
|
}
|
|
}
|
|
|
|
private:
|
|
Eigen::Vector2f origin_;
|
|
};
|
|
|
|
float ComputeRangeWeightFactor(float range, int exponent) {
|
|
float weight = 0.f;
|
|
if (std::abs(range) > kMinRangeMeters) {
|
|
weight = 1.f / (std::pow(range, exponent));
|
|
}
|
|
return weight;
|
|
}
|
|
} // namespace
|
|
|
|
proto::TSDFRangeDataInserterOptions2D CreateTSDFRangeDataInserterOptions2D(
|
|
common::LuaParameterDictionary* parameter_dictionary) {
|
|
proto::TSDFRangeDataInserterOptions2D options;
|
|
options.set_truncation_distance(
|
|
parameter_dictionary->GetDouble("truncation_distance"));
|
|
options.set_maximum_weight(parameter_dictionary->GetDouble("maximum_weight"));
|
|
options.set_update_free_space(
|
|
parameter_dictionary->GetBool("update_free_space"));
|
|
*options
|
|
.mutable_normal_estimation_options() = CreateNormalEstimationOptions2D(
|
|
parameter_dictionary->GetDictionary("normal_estimation_options").get());
|
|
options.set_project_sdf_distance_to_scan_normal(
|
|
parameter_dictionary->GetBool("project_sdf_distance_to_scan_normal"));
|
|
options.set_update_weight_range_exponent(
|
|
parameter_dictionary->GetInt("update_weight_range_exponent"));
|
|
options.set_update_weight_angle_scan_normal_to_ray_kernel_bandwidth(
|
|
parameter_dictionary->GetDouble(
|
|
"update_weight_angle_scan_normal_to_ray_kernel_bandwidth"));
|
|
options.set_update_weight_distance_cell_to_hit_kernel_bandwidth(
|
|
parameter_dictionary->GetDouble(
|
|
"update_weight_distance_cell_to_hit_kernel_bandwidth"));
|
|
return options;
|
|
}
|
|
|
|
TSDFRangeDataInserter2D::TSDFRangeDataInserter2D(
|
|
const proto::TSDFRangeDataInserterOptions2D& options)
|
|
: options_(options) {}
|
|
|
|
// Casts a ray from origin towards hit for each hit in range data.
|
|
// If 'options.update_free_space' is 'true', all cells along the ray
|
|
// until 'truncation_distance' behind hit are updated. Otherwise, only the cells
|
|
// within 'truncation_distance' around hit are updated.
|
|
void TSDFRangeDataInserter2D::Insert(const sensor::RangeData& range_data,
|
|
GridInterface* grid) const {
|
|
const float truncation_distance =
|
|
static_cast<float>(options_.truncation_distance());
|
|
TSDF2D* tsdf = static_cast<TSDF2D*>(grid);
|
|
GrowAsNeeded(range_data, truncation_distance, tsdf);
|
|
|
|
// Compute normals if needed.
|
|
bool scale_update_weight_angle_scan_normal_to_ray =
|
|
options_.update_weight_angle_scan_normal_to_ray_kernel_bandwidth() != 0.f;
|
|
sensor::RangeData sorted_range_data = range_data;
|
|
std::vector<float> normals;
|
|
if (options_.project_sdf_distance_to_scan_normal() ||
|
|
scale_update_weight_angle_scan_normal_to_ray) {
|
|
std::vector<sensor::RangefinderPoint> returns =
|
|
sorted_range_data.returns.points();
|
|
std::sort(returns.begin(), returns.end(),
|
|
RangeDataSorter(sorted_range_data.origin));
|
|
sorted_range_data.returns = sensor::PointCloud(std::move(returns));
|
|
normals = EstimateNormals(sorted_range_data,
|
|
options_.normal_estimation_options());
|
|
}
|
|
|
|
const Eigen::Vector2f origin = sorted_range_data.origin.head<2>();
|
|
for (size_t hit_index = 0; hit_index < sorted_range_data.returns.size();
|
|
++hit_index) {
|
|
const Eigen::Vector2f hit =
|
|
sorted_range_data.returns[hit_index].position.head<2>();
|
|
const float normal = normals.empty()
|
|
? std::numeric_limits<float>::quiet_NaN()
|
|
: normals[hit_index];
|
|
InsertHit(options_, hit, origin, normal, tsdf);
|
|
}
|
|
tsdf->FinishUpdate();
|
|
}
|
|
|
|
void TSDFRangeDataInserter2D::InsertHit(
|
|
const proto::TSDFRangeDataInserterOptions2D& options,
|
|
const Eigen::Vector2f& hit, const Eigen::Vector2f& origin, float normal,
|
|
TSDF2D* tsdf) const {
|
|
const Eigen::Vector2f ray = hit - origin;
|
|
const float range = ray.norm();
|
|
const float truncation_distance =
|
|
static_cast<float>(options_.truncation_distance());
|
|
if (range < truncation_distance) return;
|
|
const float truncation_ratio = truncation_distance / range;
|
|
const Eigen::Vector2f ray_begin =
|
|
options_.update_free_space() ? origin
|
|
: origin + (1.0f - truncation_ratio) * ray;
|
|
const Eigen::Vector2f ray_end = origin + (1.0f + truncation_ratio) * ray;
|
|
std::pair<Eigen::Array2i, Eigen::Array2i> superscaled_ray =
|
|
SuperscaleRay(ray_begin, ray_end, tsdf);
|
|
std::vector<Eigen::Array2i> ray_mask = RayToPixelMask(
|
|
superscaled_ray.first, superscaled_ray.second, kSubpixelScale);
|
|
|
|
// Precompute weight factors.
|
|
float weight_factor_angle_ray_normal = 1.f;
|
|
if (options_.update_weight_angle_scan_normal_to_ray_kernel_bandwidth() !=
|
|
0.f) {
|
|
const Eigen::Vector2f negative_ray = -ray;
|
|
float angle_ray_normal =
|
|
common::NormalizeAngleDifference(normal - common::atan2(negative_ray));
|
|
weight_factor_angle_ray_normal = GaussianKernel(
|
|
angle_ray_normal,
|
|
options_.update_weight_angle_scan_normal_to_ray_kernel_bandwidth());
|
|
}
|
|
float weight_factor_range = 1.f;
|
|
if (options_.update_weight_range_exponent() != 0) {
|
|
weight_factor_range = ComputeRangeWeightFactor(
|
|
range, options_.update_weight_range_exponent());
|
|
}
|
|
|
|
// Update Cells.
|
|
for (const Eigen::Array2i& cell_index : ray_mask) {
|
|
if (tsdf->CellIsUpdated(cell_index)) continue;
|
|
Eigen::Vector2f cell_center = tsdf->limits().GetCellCenter(cell_index);
|
|
float distance_cell_to_origin = (cell_center - origin).norm();
|
|
float update_tsd = range - distance_cell_to_origin;
|
|
if (options_.project_sdf_distance_to_scan_normal()) {
|
|
float normal_orientation = normal;
|
|
update_tsd = (cell_center - hit)
|
|
.dot(Eigen::Vector2f{std::cos(normal_orientation),
|
|
std::sin(normal_orientation)});
|
|
}
|
|
update_tsd =
|
|
common::Clamp(update_tsd, -truncation_distance, truncation_distance);
|
|
float update_weight = weight_factor_range * weight_factor_angle_ray_normal;
|
|
if (options_.update_weight_distance_cell_to_hit_kernel_bandwidth() != 0.f) {
|
|
update_weight *= GaussianKernel(
|
|
update_tsd,
|
|
options_.update_weight_distance_cell_to_hit_kernel_bandwidth());
|
|
}
|
|
UpdateCell(cell_index, update_tsd, update_weight, tsdf);
|
|
}
|
|
}
|
|
|
|
void TSDFRangeDataInserter2D::UpdateCell(const Eigen::Array2i& cell,
|
|
float update_sdf, float update_weight,
|
|
TSDF2D* tsdf) const {
|
|
if (update_weight == 0.f) return;
|
|
const std::pair<float, float> tsd_and_weight = tsdf->GetTSDAndWeight(cell);
|
|
float updated_weight = tsd_and_weight.second + update_weight;
|
|
float updated_sdf = (tsd_and_weight.first * tsd_and_weight.second +
|
|
update_sdf * update_weight) /
|
|
updated_weight;
|
|
updated_weight =
|
|
std::min(updated_weight, static_cast<float>(options_.maximum_weight()));
|
|
tsdf->SetCell(cell, updated_sdf, updated_weight);
|
|
}
|
|
|
|
} // namespace mapping
|
|
} // namespace cartographer
|