Adds probability grid serialization to proto (#208)
							parent
							
								
									30b9fcce34
								
							
						
					
					
						commit
						8bf6f101c5
					
				|  | @ -23,8 +23,9 @@ | |||
| #include "Eigen/Core" | ||||
| #include "Eigen/Geometry" | ||||
| #include "cartographer/common/math.h" | ||||
| #include "cartographer/mapping/trajectory_node.h" | ||||
| #include "cartographer/mapping_2d/proto/map_limits.pb.h" | ||||
| #include "cartographer/mapping_2d/xy_index.h" | ||||
| #include "cartographer/mapping/trajectory_node.h" | ||||
| #include "cartographer/sensor/laser.h" | ||||
| #include "cartographer/sensor/point_cloud.h" | ||||
| #include "cartographer/transform/rigid_transform.h" | ||||
|  | @ -46,6 +47,11 @@ class MapLimits { | |||
|     CHECK_GT(cell_limits.num_y_cells, 0.); | ||||
|   } | ||||
| 
 | ||||
|   explicit MapLimits(const proto::MapLimits& map_limits) | ||||
|       : resolution_(map_limits.resolution()), | ||||
|         max_(transform::ToEigen(map_limits.max())), | ||||
|         cell_limits_(map_limits.cell_limits()) {} | ||||
| 
 | ||||
|   // Returns the cell size in meters. All cells are square and the resolution is
 | ||||
|   // the length of one side.
 | ||||
|   double resolution() const { return resolution_; } | ||||
|  | @ -73,9 +79,8 @@ class MapLimits { | |||
|   // Returns true of the ProbabilityGrid contains 'xy_index'.
 | ||||
|   bool Contains(const Eigen::Array2i& xy_index) const { | ||||
|     return (Eigen::Array2i(0, 0) <= xy_index).all() && | ||||
|            (xy_index < | ||||
|             Eigen::Array2i(cell_limits_.num_x_cells, cell_limits_.num_y_cells)) | ||||
|                .all(); | ||||
|            (xy_index < Eigen::Array2i(cell_limits_.num_x_cells, | ||||
|                                       cell_limits_.num_y_cells)).all(); | ||||
|   } | ||||
| 
 | ||||
|   // Computes MapLimits that contain the origin, and all laser rays (both
 | ||||
|  | @ -129,6 +134,14 @@ class MapLimits { | |||
|   CellLimits cell_limits_; | ||||
| }; | ||||
| 
 | ||||
| inline proto::MapLimits ToProto(const MapLimits& map_limits) { | ||||
|   proto::MapLimits result; | ||||
|   result.set_resolution(map_limits.resolution()); | ||||
|   *result.mutable_max() = transform::ToProto(map_limits.max()); | ||||
|   *result.mutable_cell_limits() = ToProto(map_limits.cell_limits()); | ||||
|   return result; | ||||
| } | ||||
| 
 | ||||
| }  // namespace mapping_2d
 | ||||
| }  // namespace cartographer
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -22,6 +22,32 @@ namespace cartographer { | |||
| namespace mapping_2d { | ||||
| namespace { | ||||
| 
 | ||||
| TEST(MapLimitsTest, ToProto) { | ||||
|   const MapLimits limits(42., Eigen::Vector2d(3., 0.), CellLimits(2, 3)); | ||||
|   const auto proto = ToProto(limits); | ||||
|   EXPECT_EQ(limits.resolution(), proto.resolution()); | ||||
|   EXPECT_EQ(limits.max().x(), proto.max().x()); | ||||
|   EXPECT_EQ(limits.max().y(), proto.max().y()); | ||||
|   EXPECT_EQ(ToProto(limits.cell_limits()).DebugString(), | ||||
|             proto.cell_limits().DebugString()); | ||||
| } | ||||
| 
 | ||||
| TEST(MapLimitsTest, ProtoConstructor) { | ||||
|   proto::MapLimits limits; | ||||
|   limits.set_resolution(1.); | ||||
|   limits.mutable_max()->set_x(2.); | ||||
|   limits.mutable_max()->set_y(3.); | ||||
|   limits.mutable_cell_limits()->set_num_x_cells(4); | ||||
|   limits.mutable_cell_limits()->set_num_y_cells(5); | ||||
| 
 | ||||
|   const MapLimits native(limits); | ||||
|   EXPECT_EQ(limits.resolution(), native.resolution()); | ||||
|   EXPECT_EQ(limits.max().x(), native.max().x()); | ||||
|   EXPECT_EQ(limits.max().y(), native.max().y()); | ||||
|   EXPECT_EQ(limits.cell_limits().DebugString(), | ||||
|             ToProto(native.cell_limits()).DebugString()); | ||||
| } | ||||
| 
 | ||||
| TEST(MapLimitsTest, ConstructAndGet) { | ||||
|   const MapLimits limits(42., Eigen::Vector2d(3., 0.), CellLimits(2, 3)); | ||||
| 
 | ||||
|  |  | |||
|  | @ -27,9 +27,10 @@ | |||
| 
 | ||||
| #include "cartographer/common/math.h" | ||||
| #include "cartographer/common/port.h" | ||||
| #include "cartographer/mapping/probability_values.h" | ||||
| #include "cartographer/mapping_2d/map_limits.h" | ||||
| #include "cartographer/mapping_2d/proto/probability_grid.pb.h" | ||||
| #include "cartographer/mapping_2d/xy_index.h" | ||||
| #include "cartographer/mapping/probability_values.h" | ||||
| #include "glog/logging.h" | ||||
| 
 | ||||
| namespace cartographer { | ||||
|  | @ -48,6 +49,22 @@ class ProbabilityGrid { | |||
|         min_x_(limits_.cell_limits().num_x_cells - 1), | ||||
|         min_y_(limits_.cell_limits().num_y_cells - 1) {} | ||||
| 
 | ||||
|   explicit ProbabilityGrid(const proto::ProbabilityGrid& proto) | ||||
|       : limits_(proto.limits()), | ||||
|         cells_(), | ||||
|         update_indices_(proto.update_indices().begin(), | ||||
|                         proto.update_indices().end()), | ||||
|         max_x_(proto.max_x()), | ||||
|         max_y_(proto.max_y()), | ||||
|         min_x_(proto.min_x()), | ||||
|         min_y_(proto.min_y()) { | ||||
|     cells_.reserve(proto.cells_size()); | ||||
|     for (const auto cell : proto.cells()) { | ||||
|       CHECK_LE(cell, std::numeric_limits<uint16>::max()); | ||||
|       cells_.push_back(cell); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   // Returns the limits of this ProbabilityGrid.
 | ||||
|   const MapLimits& limits() const { return limits_; } | ||||
| 
 | ||||
|  | @ -106,8 +123,9 @@ class ProbabilityGrid { | |||
| 
 | ||||
|   // Returns true if the probability at the specified index is known.
 | ||||
|   bool IsKnown(const Eigen::Array2i& xy_index) const { | ||||
|     return limits_.Contains(xy_index) && cells_[GetIndexOfCell(xy_index)] != | ||||
|                                              mapping::kUnknownProbabilityValue; | ||||
|     return limits_.Contains(xy_index) && | ||||
|            cells_[GetIndexOfCell(xy_index)] != | ||||
|                mapping::kUnknownProbabilityValue; | ||||
|   } | ||||
| 
 | ||||
|   // Fills in 'offset' and 'limits' to define a subregion of that contains all
 | ||||
|  | @ -154,6 +172,24 @@ class ProbabilityGrid { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   proto::ProbabilityGrid ToProto() { | ||||
|     proto::ProbabilityGrid result; | ||||
|     *result.mutable_limits() = cartographer::mapping_2d::ToProto(limits_); | ||||
|     result.mutable_cells()->Reserve(cells_.size()); | ||||
|     for (const auto cell : cells_) { | ||||
|       result.mutable_cells()->Add(cell); | ||||
|     } | ||||
|     result.mutable_update_indices()->Reserve(update_indices_.size()); | ||||
|     for (const auto update : update_indices_) { | ||||
|       result.mutable_update_indices()->Add(update); | ||||
|     } | ||||
|     result.set_max_x(max_x_); | ||||
|     result.set_max_y(max_y_); | ||||
|     result.set_min_x(min_x_); | ||||
|     result.set_min_y(min_y_); | ||||
|     return result; | ||||
|   } | ||||
| 
 | ||||
|  private: | ||||
|   // Returns the index of the cell at 'xy_index'.
 | ||||
|   int GetIndexOfCell(const Eigen::Array2i& xy_index) const { | ||||
|  | @ -172,7 +208,7 @@ class ProbabilityGrid { | |||
|   std::vector<uint16> cells_;  // Highest bit is update marker.
 | ||||
|   std::vector<int> update_indices_; | ||||
| 
 | ||||
|   // Minimum and maximum cell coordinates of know cells to efficiently compute
 | ||||
|   // Minimum and maximum cell coordinates of known cells to efficiently compute
 | ||||
|   // cropping limits.
 | ||||
|   int max_x_; | ||||
|   int max_y_; | ||||
|  |  | |||
|  | @ -17,6 +17,7 @@ | |||
| #include "cartographer/mapping_2d/probability_grid.h" | ||||
| 
 | ||||
| #include <random> | ||||
| #include <vector> | ||||
| 
 | ||||
| #include "gtest/gtest.h" | ||||
| 
 | ||||
|  | @ -24,6 +25,40 @@ namespace cartographer { | |||
| namespace mapping_2d { | ||||
| namespace { | ||||
| 
 | ||||
| TEST(ProbabilityGridTest, ProtoConstructor) { | ||||
|   proto::ProbabilityGrid proto; | ||||
|   const MapLimits limits(1., {2., 3.}, CellLimits(4., 5.)); | ||||
|   *proto.mutable_limits() = ToProto(limits); | ||||
|   for (int i = 6; i < 12; ++i) { | ||||
|     proto.mutable_cells()->Add(static_cast<uint16>(i)); | ||||
|   } | ||||
|   for (int i = 13; i < 18; ++i) { | ||||
|     proto.mutable_update_indices()->Add(i); | ||||
|   } | ||||
|   proto.set_max_x(19); | ||||
|   proto.set_max_y(20); | ||||
|   proto.set_min_x(21); | ||||
|   proto.set_min_y(22); | ||||
| 
 | ||||
|   ProbabilityGrid grid(proto); | ||||
|   EXPECT_EQ(proto.limits().DebugString(), ToProto(grid.limits()).DebugString()); | ||||
| 
 | ||||
|   // TODO(macmason): Figure out how to test the contents of cells_,
 | ||||
|   // update_indices_, and {min, max}_{x, y}_ gracefully.
 | ||||
| } | ||||
| 
 | ||||
| TEST(ProbabilityGridTest, ToProto) { | ||||
|   ProbabilityGrid probability_grid( | ||||
|       MapLimits(1., Eigen::Vector2d(1., 1.), CellLimits(2, 2))); | ||||
| 
 | ||||
|   const auto proto = probability_grid.ToProto(); | ||||
|   EXPECT_EQ(ToProto(probability_grid.limits()).DebugString(), | ||||
|             proto.limits().DebugString()); | ||||
| 
 | ||||
|   // TODO(macmason): Figure out how to test the contents of cells_,
 | ||||
|   // update_indices_, and {min, max}_{x, y}_ gracefully.
 | ||||
| } | ||||
| 
 | ||||
| TEST(ProbabilityGridTest, ApplyOdds) { | ||||
|   ProbabilityGrid probability_grid( | ||||
|       MapLimits(1., Eigen::Vector2d(1., 1.), CellLimits(2, 2))); | ||||
|  | @ -111,35 +146,26 @@ TEST(ProbabilityGridTest, GetXYIndexOfCellContainingPoint) { | |||
|   const CellLimits& cell_limits = limits.cell_limits(); | ||||
|   ASSERT_EQ(14, cell_limits.num_x_cells); | ||||
|   ASSERT_EQ(8, cell_limits.num_y_cells); | ||||
|   EXPECT_TRUE( | ||||
|       (Eigen::Array2i(0, 0) == limits.GetXYIndexOfCellContainingPoint(7, 13)) | ||||
|           .all()); | ||||
|   EXPECT_TRUE( | ||||
|       (Eigen::Array2i(13, 0) == limits.GetXYIndexOfCellContainingPoint(7, -13)) | ||||
|           .all()); | ||||
|   EXPECT_TRUE( | ||||
|       (Eigen::Array2i(0, 7) == limits.GetXYIndexOfCellContainingPoint(-7, 13)) | ||||
|           .all()); | ||||
|   EXPECT_TRUE( | ||||
|       (Eigen::Array2i(13, 7) == limits.GetXYIndexOfCellContainingPoint(-7, -13)) | ||||
|           .all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(0, 0) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(7, 13)).all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(13, 0) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(7, -13)).all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(0, 7) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(-7, 13)).all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(13, 7) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(-7, -13)).all()); | ||||
| 
 | ||||
|   // Check around the origin.
 | ||||
|   EXPECT_TRUE( | ||||
|       (Eigen::Array2i(6, 3) == limits.GetXYIndexOfCellContainingPoint(0.5, 0.5)) | ||||
|           .all()); | ||||
|   EXPECT_TRUE( | ||||
|       (Eigen::Array2i(6, 3) == limits.GetXYIndexOfCellContainingPoint(1.5, 1.5)) | ||||
|           .all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(6, 3) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(0.5, 0.5)).all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(6, 3) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(1.5, 1.5)).all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(7, 3) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(0.5, -0.5)) | ||||
|                   .all()); | ||||
|                limits.GetXYIndexOfCellContainingPoint(0.5, -0.5)).all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(6, 4) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(-0.5, 0.5)) | ||||
|                   .all()); | ||||
|                limits.GetXYIndexOfCellContainingPoint(-0.5, 0.5)).all()); | ||||
|   EXPECT_TRUE((Eigen::Array2i(7, 4) == | ||||
|                limits.GetXYIndexOfCellContainingPoint(-0.5, -0.5)) | ||||
|                   .all()); | ||||
|                limits.GetXYIndexOfCellContainingPoint(-0.5, -0.5)).all()); | ||||
| } | ||||
| 
 | ||||
| TEST(ProbabilityGridTest, CorrectCropping) { | ||||
|  |  | |||
|  | @ -0,0 +1,22 @@ | |||
| // 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. | ||||
| 
 | ||||
| syntax = "proto2"; | ||||
| 
 | ||||
| package cartographer.mapping_2d.proto; | ||||
| 
 | ||||
| message CellLimits { | ||||
|   optional int32 num_x_cells = 1; | ||||
|   optional int32 num_y_cells = 2; | ||||
| } | ||||
|  | @ -0,0 +1,26 @@ | |||
| // 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. | ||||
| 
 | ||||
| syntax = "proto2"; | ||||
| 
 | ||||
| import "cartographer/mapping_2d/proto/cell_limits.proto"; | ||||
| import "cartographer/transform/proto/transform.proto"; | ||||
| 
 | ||||
| package cartographer.mapping_2d.proto; | ||||
| 
 | ||||
| message MapLimits { | ||||
|   optional double resolution = 1; | ||||
|   optional cartographer.transform.proto.Vector2d max = 2; | ||||
|   optional CellLimits cell_limits = 3; | ||||
| } | ||||
|  | @ -0,0 +1,31 @@ | |||
| // 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. | ||||
| 
 | ||||
| syntax = "proto2"; | ||||
| 
 | ||||
| import "cartographer/mapping_2d/proto/map_limits.proto"; | ||||
| 
 | ||||
| package cartographer.mapping_2d.proto; | ||||
| 
 | ||||
| message ProbabilityGrid { | ||||
|   optional MapLimits limits = 1; | ||||
|   // These values are actually int16s, but protos don't have a native int16 type. | ||||
|   repeated int32 cells = 2; | ||||
|   repeated int32 update_indices = 3; | ||||
|   optional int32 max_x = 4; | ||||
|   optional int32 max_y = 5; | ||||
|   optional int32 min_x = 6; | ||||
|   optional int32 min_y = 7; | ||||
| } | ||||
| 
 | ||||
|  | @ -25,6 +25,7 @@ | |||
| #include "Eigen/Core" | ||||
| #include "cartographer/common/math.h" | ||||
| #include "cartographer/common/port.h" | ||||
| #include "cartographer/mapping_2d/proto/cell_limits.pb.h" | ||||
| #include "glog/logging.h" | ||||
| 
 | ||||
| namespace cartographer { | ||||
|  | @ -35,10 +36,21 @@ struct CellLimits { | |||
|   CellLimits(int init_num_x_cells, int init_num_y_cells) | ||||
|       : num_x_cells(init_num_x_cells), num_y_cells(init_num_y_cells) {} | ||||
| 
 | ||||
|   explicit CellLimits(const proto::CellLimits& cell_limits) | ||||
|       : num_x_cells(cell_limits.num_x_cells()), | ||||
|         num_y_cells(cell_limits.num_y_cells()) {} | ||||
| 
 | ||||
|   int num_x_cells = 0; | ||||
|   int num_y_cells = 0; | ||||
| }; | ||||
| 
 | ||||
| inline proto::CellLimits ToProto(const CellLimits& cell_limits) { | ||||
|   proto::CellLimits result; | ||||
|   result.set_num_x_cells(cell_limits.num_x_cells); | ||||
|   result.set_num_y_cells(cell_limits.num_y_cells); | ||||
|   return result; | ||||
| } | ||||
| 
 | ||||
| // Iterates in row-major order through a range of xy-indices.
 | ||||
| class XYIndexRangeIterator | ||||
|     : public std::iterator<std::input_iterator_tag, Eigen::Array2i> { | ||||
|  |  | |||
|  | @ -22,6 +22,23 @@ namespace cartographer { | |||
| namespace mapping_2d { | ||||
| namespace { | ||||
| 
 | ||||
| TEST(XYIndexTest, CellLimitsToProto) { | ||||
|   const CellLimits limits(1, 2); | ||||
|   const auto proto = ToProto(limits); | ||||
|   EXPECT_EQ(limits.num_x_cells, proto.num_x_cells()); | ||||
|   EXPECT_EQ(limits.num_y_cells, proto.num_y_cells()); | ||||
| } | ||||
| 
 | ||||
| TEST(XYIndexTest, CellLimitsProtoConstructor) { | ||||
|   proto::CellLimits limits; | ||||
|   limits.set_num_x_cells(1); | ||||
|   limits.set_num_y_cells(2); | ||||
| 
 | ||||
|   auto native = CellLimits(limits); | ||||
|   EXPECT_EQ(limits.num_x_cells(), native.num_x_cells); | ||||
|   EXPECT_EQ(limits.num_y_cells(), native.num_y_cells); | ||||
| } | ||||
| 
 | ||||
| TEST(XYIndexTest, XYIndexRangeIterator) { | ||||
|   const Eigen::Array2i min(1, 2); | ||||
|   const Eigen::Array2i max(3, 4); | ||||
|  |  | |||
|  | @ -76,6 +76,13 @@ proto::Rigid3f ToProto(const transform::Rigid3f& rigid) { | |||
|   return proto; | ||||
| } | ||||
| 
 | ||||
| proto::Vector2d ToProto(const Eigen::Vector2d& vector) { | ||||
|   proto::Vector2d proto; | ||||
|   proto.set_x(vector.x()); | ||||
|   proto.set_y(vector.y()); | ||||
|   return proto; | ||||
| } | ||||
| 
 | ||||
| proto::Vector3f ToProto(const Eigen::Vector3f& vector) { | ||||
|   proto::Vector3f proto; | ||||
|   proto.set_x(vector.x()); | ||||
|  |  | |||
|  | @ -125,6 +125,7 @@ proto::Rigid2f ToProto(const Rigid2f& transform); | |||
| proto::Rigid3d ToProto(const Rigid3d& rigid); | ||||
| Rigid3d ToRigid3(const proto::Rigid3d& rigid); | ||||
| proto::Rigid3f ToProto(const Rigid3f& rigid); | ||||
| proto::Vector2d ToProto(const Eigen::Vector2d& vector); | ||||
| proto::Vector3f ToProto(const Eigen::Vector3f& vector); | ||||
| proto::Vector3d ToProto(const Eigen::Vector3d& vector); | ||||
| proto::Quaternionf ToProto(const Eigen::Quaternionf& quaternion); | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue