From 033be5eddd45c97f46178760fd719223363cf1e5 Mon Sep 17 00:00:00 2001 From: chris7462 Date: Sun, 6 Nov 2022 20:33:54 -0500 Subject: [PATCH] ROS2 migration --- CMakeLists.txt | 142 +++++++--- README.md => ReadMe.md | 85 +++--- include/floam/laser_mapping_class.hpp | 49 ++++ include/floam/laser_mapping_node.hpp | 40 +++ include/floam/laser_processing_class.hpp | 47 ++++ include/floam/laser_processing_node.hpp | 40 +++ include/floam/lidar.hpp | 31 +++ .../lidar_optimization.hpp} | 45 ++- .../odom_estimation_class.hpp} | 59 ++-- include/floam/odom_estimation_node.hpp | 47 ++++ include/laserMappingClass.h | 67 ----- include/laserProcessingClass.h | 48 ---- include/lidar.h | 40 --- launch/floam.launch | 50 ---- launch/floam.launch.py | 51 ++++ launch/floam_mapping.launch | 52 ---- launch/floam_mapping.launch.py | 59 ++++ launch/floam_velodyne.launch | 45 --- package.xml | 47 ++-- params/floam_params.yaml | 25 ++ rviz/floam.rviz | 192 ++++++------- rviz/floam_mapping.rviz | 228 +++++++-------- rviz/floam_octomap.rviz | 233 ++++++++++++++++ rviz/floam_velodyne.rviz | 233 ---------------- src/laserMappingClass.cpp | 205 -------------- src/laserMappingNode.cpp | 137 --------- src/laserProcessingClass.cpp | 224 --------------- src/laserProcessingNode.cpp | 143 ---------- src/laser_mapping.cpp | 120 ++++++++ src/laser_mapping_class.cpp | 210 ++++++++++++++ src/laser_mapping_node.cpp | 23 ++ src/laser_processing.cpp | 111 ++++++++ src/laser_processing_class.cpp | 167 +++++++++++ src/laser_processing_node.cpp | 23 ++ src/lidar.cpp | 47 ++-- src/lidarOptimization.cpp | 152 ---------- src/lidar_optimization.cpp | 141 ++++++++++ src/odomEstimationClass.cpp | 260 ------------------ src/odomEstimationNode.cpp | 167 ----------- src/odom_estimation.cpp | 165 +++++++++++ src/odom_estimation_class.cpp | 258 +++++++++++++++++ src/odom_estimation_node.cpp | 23 ++ 42 files changed, 2257 insertions(+), 2274 deletions(-) rename README.md => ReadMe.md (62%) create mode 100644 include/floam/laser_mapping_class.hpp create mode 100644 include/floam/laser_mapping_node.hpp create mode 100644 include/floam/laser_processing_class.hpp create mode 100644 include/floam/laser_processing_node.hpp create mode 100644 include/floam/lidar.hpp rename include/{lidarOptimization.h => floam/lidar_optimization.hpp} (61%) rename include/{odomEstimationClass.h => floam/odom_estimation_class.hpp} (71%) create mode 100644 include/floam/odom_estimation_node.hpp delete mode 100644 include/laserMappingClass.h delete mode 100644 include/laserProcessingClass.h delete mode 100644 include/lidar.h delete mode 100644 launch/floam.launch create mode 100644 launch/floam.launch.py delete mode 100644 launch/floam_mapping.launch create mode 100644 launch/floam_mapping.launch.py delete mode 100644 launch/floam_velodyne.launch create mode 100644 params/floam_params.yaml create mode 100644 rviz/floam_octomap.rviz delete mode 100644 rviz/floam_velodyne.rviz delete mode 100644 src/laserMappingClass.cpp delete mode 100644 src/laserMappingNode.cpp delete mode 100644 src/laserProcessingClass.cpp delete mode 100644 src/laserProcessingNode.cpp create mode 100644 src/laser_mapping.cpp create mode 100644 src/laser_mapping_class.cpp create mode 100644 src/laser_mapping_node.cpp create mode 100644 src/laser_processing.cpp create mode 100644 src/laser_processing_class.cpp create mode 100644 src/laser_processing_node.cpp delete mode 100644 src/lidarOptimization.cpp create mode 100644 src/lidar_optimization.cpp delete mode 100644 src/odomEstimationClass.cpp delete mode 100644 src/odomEstimationNode.cpp create mode 100644 src/odom_estimation.cpp create mode 100644 src/odom_estimation_class.cpp create mode 100644 src/odom_estimation_node.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 95b92f1..b54ce6c 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -1,65 +1,133 @@ -cmake_minimum_required(VERSION 2.8.3) +cmake_minimum_required(VERSION 3.5) project(floam) set(CMAKE_BUILD_TYPE "Release") -set(CMAKE_CXX_FLAGS "-std=c++11") -set(CMAKE_CXX_FLAGS_RELEASE "-O3 -Wall -g") +set(CMAKE_CXX_FLAGS_RELEASE "-O3 -Wall -g -pthread") -find_package(catkin REQUIRED COMPONENTS - geometry_msgs - nav_msgs - sensor_msgs - roscpp - rospy - rosbag - std_msgs - tf - eigen_conversions -) +# Default to C++14 +if(NOT CMAKE_CXX_STANDARD) + set(CMAKE_CXX_STANDARD 14) +endif() -find_package(Eigen3) -if(NOT EIGEN3_FOUND) - # Fallback to cmake_modules - find_package(cmake_modules REQUIRED) - find_package(Eigen REQUIRED) - set(EIGEN3_INCLUDE_DIRS ${EIGEN_INCLUDE_DIRS}) - set(EIGEN3_LIBRARIES ${EIGEN_LIBRARIES}) # Not strictly necessary as Eigen is head only - # Possibly map additional variables to the EIGEN3_ prefix. -else() - set(EIGEN3_INCLUDE_DIRS ${EIGEN3_INCLUDE_DIR}) +if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang") + add_compile_options(-Wall -Wextra -Wpedantic) endif() + +# find dependencies +find_package(ament_cmake REQUIRED) +# uncomment the following section in order to fill in +# further dependencies manually. +# find_package( REQUIRED) +find_package(rclcpp REQUIRED) +find_package(sensor_msgs REQUIRED) +find_package(nav_msgs REQUIRED) +find_package(geometry_msgs REQUIRED) +find_package(tf2 REQUIRED) +find_package(tf2_ros REQUIRED) +find_package(pcl_conversions REQUIRED) find_package(PCL REQUIRED) +find_package(Eigen3 REQUIRED) find_package(Ceres REQUIRED) include_directories( include - ${catkin_INCLUDE_DIRS} ${PCL_INCLUDE_DIRS} + ${EIGEN3_INCLUDE_DIR} ${CERES_INCLUDE_DIRS} - ${EIGEN3_INCLUDE_DIRS} ) link_directories( - include ${PCL_LIBRARY_DIRS} ${CERES_LIBRARY_DIRS} ) +add_executable(laser_processing_node + src/laser_processing_node.cpp + src/laser_processing.cpp + src/laser_processing_class.cpp + src/lidar.cpp +) -catkin_package( - CATKIN_DEPENDS geometry_msgs nav_msgs roscpp rospy std_msgs - DEPENDS EIGEN3 PCL Ceres - INCLUDE_DIRS include +ament_target_dependencies(laser_processing_node + rclcpp + sensor_msgs + pcl_conversions ) +target_link_libraries(laser_processing_node + ${PCL_LIBRARIES} +) +add_executable(odom_estimation_node + src/odom_estimation_node.cpp + src/odom_estimation.cpp + src/odom_estimation_class.cpp + src/lidar_optimization.cpp + src/lidar.cpp +) -add_executable(floam_laser_processing_node src/laserProcessingNode.cpp src/laserProcessingClass.cpp src/lidar.cpp) -target_link_libraries(floam_laser_processing_node ${EIGEN3_LIBRARIES} ${catkin_LIBRARIES} ${PCL_LIBRARIES} ${CERES_LIBRARIES}) +ament_target_dependencies(odom_estimation_node + rclcpp + sensor_msgs + nav_msgs + geometry_msgs + tf2 + tf2_ros + pcl_conversions + Eigen3 + #CERES +) -add_executable(floam_odom_estimation_node src/odomEstimationNode.cpp src/lidarOptimization.cpp src/lidar.cpp src/odomEstimationClass.cpp) -target_link_libraries(floam_odom_estimation_node ${EIGEN3_LIBRARIES} ${catkin_LIBRARIES} ${PCL_LIBRARIES} ${CERES_LIBRARIES}) +target_link_libraries(odom_estimation_node + ${PCL_LIBRARIES} + #${EIGEN3_LIBRARIES} + ${CERES_LIBRARIES} +) -add_executable(floam_laser_mapping_node src/laserMappingNode.cpp src/laserMappingClass.cpp src/lidar.cpp) -target_link_libraries(floam_laser_mapping_node ${EIGEN3_LIBRARIES} ${catkin_LIBRARIES} ${PCL_LIBRARIES} ${CERES_LIBRARIES}) +add_executable(laser_mapping_node + src/laser_mapping_node.cpp + src/laser_mapping.cpp + src/laser_mapping_class.cpp + src/lidar.cpp +) + +ament_target_dependencies(laser_mapping_node + rclcpp + sensor_msgs + nav_msgs + pcl_conversions + Eigen3 +) + +target_link_libraries(laser_mapping_node + ${PCL_LIBRARIES} +) + +install(TARGETS laser_processing_node odom_estimation_node laser_mapping_node + DESTINATION lib/${PROJECT_NAME} +) + +install(DIRECTORY launch + DESTINATION share/${PROJECT_NAME} +) + +install(DIRECTORY rviz + DESTINATION share/${PROJECT_NAME} +) + +install(DIRECTORY params + DESTINATION share/${PROJECT_NAME} +) + +if(BUILD_TESTING) + find_package(ament_lint_auto REQUIRED) + # the following line skips the linter which checks for copyrights + # uncomment the line when a copyright and license is not present in all source files + #set(ament_cmake_copyright_FOUND TRUE) + # the following line skips cpplint (only works in a git repo) + # uncomment the line when this package is not in a git repo + #set(ament_cmake_cpplint_FOUND TRUE) + ament_lint_auto_find_test_dependencies() +endif() +ament_package() diff --git a/README.md b/ReadMe.md similarity index 62% rename from README.md rename to ReadMe.md index 181d7fe..77561fa 100644 --- a/README.md +++ b/ReadMe.md @@ -1,4 +1,4 @@ -# FLOAM +# FLOAM ## Fast LOAM (Lidar Odometry And Mapping) This work is an optimized version of A-LOAM and LOAM with the computational cost reduced by up to 3 times. @@ -6,6 +6,8 @@ This code is modified from [LOAM](https://github.com/laboshinl/loam_velodyne) an **Modifier:** [Wang Han](http://wanghan.pro), Nanyang Technological University, Singapore +**ROS2 Migration:** [Yi-Chen Zhang](chris7462.github.io/), Isuzu Technical Center of America, USA + ## 1. Demo Highlights Watch our demo at [Video Link](https://youtu.be/PzZly1SQtng)

@@ -17,7 +19,7 @@ Watch our demo at [Video Link](https://youtu.be/PzZly1SQtng) ## 2. Evaluation ### 2.1. Computational efficiency evaluation Computational efficiency evaluation (based on KITTI dataset): -Platform: Intel® Core™ i7-8700 CPU @ 3.20GHz +Platform: Intel® Core™ i7-8700 CPU @ 3.20GHz | Dataset | ALOAM | FLOAM | |----------------------------------------------|----------------------------|------------------------| | `KITTI` | 151ms | 59ms | @@ -43,85 +45,68 @@ Localization error: ## 3. Prerequisites ### 3.1 **Ubuntu** and **ROS** -Ubuntu 64-bit 18.04. +Ubuntu 64-bit 20.04. -ROS Melodic. [ROS Installation](http://wiki.ros.org/ROS/Installation) +ROS2 Foxy. [ROS Installation](https://docs.ros.org/en/foxy/Installation.html) ### 3.2. **Ceres Solver** Follow [Ceres Installation](http://ceres-solver.org/installation.html). +Please checkout to 2.0.0 tag. ROS2 migrated version doest not support Ceres 2.1.0 yet. ### 3.3. **PCL** -Follow [PCL Installation](http://www.pointclouds.org/downloads/linux.html). - -### 3.4. **Trajectory visualization** -For visualization purpose, this package uses hector trajectory sever, you may install the package by +For PCL library, please install by the following: +```bash +sudo apt-get install libpcl-dev ros-foxy-pcl-ros ``` -sudo apt-get install ros-melodic-hector-trajectory-server -``` -Alternatively, you may remove the hector trajectory server node if trajectory visualization is not needed -## 4. Build +## 4. Build ### 4.1 Clone repository: -``` - cd ~/catkin_ws/src - git clone https://github.com/wh200720041/floam.git - cd .. - catkin_make - source ~/catkin_ws/devel/setup.bash +```bash +cd ~/colcon_ws/src +git clone https://github.com/chris7462/floam.git +cd .. +colcon build +source ./install/setup.bash ``` ### 4.2 Download test rosbag Download [KITTI sequence 05](https://drive.google.com/file/d/1eyO0Io3lX2z-yYsfGHawMKZa5Z0uYJ0W/view?usp=sharing) or [KITTI sequence 07](https://drive.google.com/file/d/1_qUfwUw88rEKitUpt1kjswv7Cv4GPs0b/view?usp=sharing) -Unzip compressed file 2011_09_30_0018.zip. If your system does not have unzip. please install unzip by +Unzip compressed file 2011_09_30_0018.zip. If your system does not have unzip. please install unzip by ``` -sudo apt-get install unzip +sudo apt-get install unzip ``` And this may take a few minutes to unzip the file +```bash +cd ~/Downloads +unzip ~/Downloads/2011_09_30_0018.zip ``` - cd ~/Downloads - unzip ~/Downloads/2011_09_30_0018.zip -``` +Then convert the ROS1 bag to ROS2 bag. See [here](https://ternaris.gitlab.io/rosbags/topics/convert.html) for reference ### 4.3 Launch ROS -``` - roslaunch floam floam.launch +```bash +ros2 launch floam floam.launch.py ``` if you would like to create the map at the same time, you can run (more cpu cost) +```bash +ros2 launch floam floam_mapping.launch.py ``` - roslaunch floam floam_mapping.launch -``` -If the mapping process is slow, you may wish to change the rosbag speed by replacing "--clock -r 0.5" with "--clock -r 0.2" in your launch file, or you can change the map publish frequency manually (default is 10 Hz) - +If the mapping process is slow, you may wish to change the rosbag speed by replacing "-r 0.5" with "-r 0.2" in your launch file, or you can change the map publish frequency manually (default is 10 Hz) ## 5. Test on other sequence -To generate rosbag file of kitti dataset, you may use the tools provided by -[kitti_to_rosbag](https://github.com/ethz-asl/kitti_to_rosbag) or [kitti2bag](https://github.com/tomas789/kitti2bag) - -## 6. Test on Velodyne VLP-16 or HDL-32 -You may wish to test FLOAM on your own platform and sensor such as VLP-16 -You can install the velodyne sensor driver by -``` -sudo apt-get install ros-melodic-velodyne-pointcloud -``` -launch floam for your own velodyne sensor -``` - roslaunch floam floam_velodyne.launch -``` -If you are using HDL-32 or other sensor, please change the scan_line in the launch file - - -## 7.Acknowledgements -Thanks for [A-LOAM](https://github.com/HKUST-Aerial-Robotics/A-LOAM) and LOAM(J. Zhang and S. Singh. LOAM: Lidar Odometry and Mapping in Real-time) and [LOAM_NOTED](https://github.com/cuitaixiang/LOAM_NOTED). +To generate rosbag file of kitti dataset, you may use the tools provided by +[kitti_to_rosbag](https://github.com/ethz-asl/kitti_to_rosbag) or [kitti2bag](https://github.com/tomas789/kitti2bag) +## 6. Acknowledgements +Thanks for [A-LOAM](https://github.com/HKUST-Aerial-Robotics/A-LOAM) and LOAM (J. Zhang and S. Singh. LOAM: Lidar Odometry and Mapping in Real-time) and [LOAM_NOTED](https://github.com/cuitaixiang/LOAM_NOTED). -## 8. Citation +## 7. Citation If you use this work for your research, you may want to cite ``` @inproceedings{wang2021, author={H. {Wang} and C. {Wang} and C. {Chen} and L. {Xie}}, - booktitle={2021 IEEE/RSJ International Conference on Intelligent Robots and Systems (IROS)}, - title={F-LOAM : Fast LiDAR Odometry and Mapping}, + booktitle={2021 IEEE/RSJ International Conference on Intelligent Robots and Systems (IROS)}, + title={F-LOAM : Fast LiDAR Odometry and Mapping}, year={2020}, volume={}, number={} diff --git a/include/floam/laser_mapping_class.hpp b/include/floam/laser_mapping_class.hpp new file mode 100644 index 0000000..09acfe1 --- /dev/null +++ b/include/floam/laser_mapping_class.hpp @@ -0,0 +1,49 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +#pragma once + +// pcl header +#include +#include +#include + +// c++ header +#include + +#define LASER_CELL_WIDTH 50.0 +#define LASER_CELL_HEIGHT 50.0 +#define LASER_CELL_DEPTH 50.0 + +// separate map as many sub point clouds + +#define LASER_CELL_RANGE_HORIZONTAL 2 +#define LASER_CELL_RANGE_VERTICAL 2 + + +class LaserMappingClass +{ + public: + LaserMappingClass() = default; + void init(double map_resolution); + void updateCurrentPointsToMap(const pcl::PointCloud::Ptr& pc_in, const Eigen::Isometry3d& pose_current); + pcl::PointCloud::Ptr getMap(); + + private: + int origin_in_map_x; + int origin_in_map_y; + int origin_in_map_z; + int map_width; + int map_height; + int map_depth; + std::vector::Ptr>>> map; + pcl::VoxelGrid downSizeFilter; + + void addWidthCellNegative(); + void addWidthCellPositive(); + void addHeightCellNegative(); + void addHeightCellPositive(); + void addDepthCellNegative(); + void addDepthCellPositive(); + void checkPoints(int& x, int& y, int& z); +}; diff --git a/include/floam/laser_mapping_node.hpp b/include/floam/laser_mapping_node.hpp new file mode 100644 index 0000000..1eb32fa --- /dev/null +++ b/include/floam/laser_mapping_node.hpp @@ -0,0 +1,40 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +#pragma once + +// ros header +#include +#include +#include + +// c++ header +#include +#include + +// local header +#include "floam/lidar.hpp" +#include "floam/laser_mapping_class.hpp" + + +class LaserMappingNode: public rclcpp::Node +{ + public: + LaserMappingNode(); + void laser_mapping(); + + private: + LaserMappingClass laserMapping_; + std::queue odometryBuf_; + std::queue pointCloudBuf_; + std::mutex mutex_lock_; + lidar::Lidar lidar_param_; + + rclcpp::Subscription::SharedPtr subLaserCloud_; + rclcpp::Subscription::SharedPtr subOdometry_; + + rclcpp::Publisher::SharedPtr pubMap_; + + void odomCallback(const nav_msgs::msg::Odometry::ConstSharedPtr odomMsg); + void velodyneHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg); +}; diff --git a/include/floam/laser_processing_class.hpp b/include/floam/laser_processing_class.hpp new file mode 100644 index 0000000..7aa3246 --- /dev/null +++ b/include/floam/laser_processing_class.hpp @@ -0,0 +1,47 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +#pragma once + +// pcl header +#include +#include + +// local header +#include "lidar.hpp" + + +// points covariance class +class Double2d +{ + public: + int id; + double value; + Double2d(int id_in, double value_in); +}; + +// points info class +class PointsInfo +{ + public: + int layer; + double time; + PointsInfo(int layer_in, double time_in); +}; + +class LaserProcessingClass +{ + public: + LaserProcessingClass() = default; + void init(lidar::Lidar lidar_param_in); + void featureExtraction(const pcl::PointCloud::Ptr& pc_in, + pcl::PointCloud::Ptr& pc_out_edge, + pcl::PointCloud::Ptr& pc_out_surf); + void featureExtractionFromSector(const pcl::PointCloud::Ptr& pc_in, + std::vector& cloudCurvature, + pcl::PointCloud::Ptr& pc_out_edge, + pcl::PointCloud::Ptr& pc_out_surf); + + private: + lidar::Lidar lidar_param; +}; diff --git a/include/floam/laser_processing_node.hpp b/include/floam/laser_processing_node.hpp new file mode 100644 index 0000000..3f44f31 --- /dev/null +++ b/include/floam/laser_processing_node.hpp @@ -0,0 +1,40 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +#pragma once + +// ros lib +#include +#include + +// c++ lib +#include +#include + +// local lib +#include "floam/lidar.hpp" +#include "floam/laser_processing_class.hpp" + + +class LaserProcessingNode: public rclcpp::Node +{ + public: + LaserProcessingNode(); + void laser_processing(); + + private: + LaserProcessingClass laserProcessing_; + std::queue pointCloudBuf_; + std::mutex mutex_lock_; + lidar::Lidar lidar_param_; + + rclcpp::Subscription::SharedPtr subLaserCloud_; + rclcpp::Publisher::SharedPtr pubEdgePoints_; + rclcpp::Publisher::SharedPtr pubSurfPoints_; + rclcpp::Publisher::SharedPtr pubLaserCloudFiltered_; + + void velodyneHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg); + + double total_time_; + int total_frame_; +}; diff --git a/include/floam/lidar.hpp b/include/floam/lidar.hpp new file mode 100644 index 0000000..1085bf7 --- /dev/null +++ b/include/floam/lidar.hpp @@ -0,0 +1,31 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +#pragma once + +namespace lidar { + +class Lidar +{ + public: + Lidar() = default; + + void setScanPeriod(double scan_period_in); + void setLines(int num_lines_in); + void setVerticalAngle(double vertical_angle_in); + void setVerticalResolution(double vertical_angle_resolution_in); + void setMaxDistance(double max_distance_in); + void setMinDistance(double min_distance_in); + + double max_distance; + double min_distance; + int num_lines; + double scan_period; + int points_per_line; + double horizontal_angle_resolution; + double horizontal_angle; + double vertical_angle_resolution; + double vertical_angle; +}; + +} diff --git a/include/lidarOptimization.h b/include/floam/lidar_optimization.hpp similarity index 61% rename from include/lidarOptimization.h rename to include/floam/lidar_optimization.hpp index 59852de..47b27b8 100644 --- a/include/lidarOptimization.h +++ b/include/floam/lidar_optimization.hpp @@ -1,44 +1,46 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro -#ifndef _LIDAR_OPTIMIZATION_ANALYTIC_H_ -#define _LIDAR_OPTIMIZATION_ANALYTIC_H_ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang +#pragma once + +// system header #include #include #include #include + void getTransformFromSe3(const Eigen::Matrix& se3, Eigen::Quaterniond& q, Eigen::Vector3d& t); Eigen::Matrix3d skew(Eigen::Vector3d& mat_in); -class EdgeAnalyticCostFunction : public ceres::SizedCostFunction<1, 7> { +class EdgeAnalyticCostFunction : public ceres::SizedCostFunction<1, 7> +{ public: - - EdgeAnalyticCostFunction(Eigen::Vector3d curr_point_, Eigen::Vector3d last_point_a_, Eigen::Vector3d last_point_b_); + EdgeAnalyticCostFunction(Eigen::Vector3d curr_point, Eigen::Vector3d last_point_a, Eigen::Vector3d last_point_b); virtual ~EdgeAnalyticCostFunction() {} virtual bool Evaluate(double const *const *parameters, double *residuals, double **jacobians) const; - Eigen::Vector3d curr_point; - Eigen::Vector3d last_point_a; - Eigen::Vector3d last_point_b; + Eigen::Vector3d curr_point_; + Eigen::Vector3d last_point_a_; + Eigen::Vector3d last_point_b_; }; -class SurfNormAnalyticCostFunction : public ceres::SizedCostFunction<1, 7> { +class SurfNormAnalyticCostFunction : public ceres::SizedCostFunction<1, 7> +{ public: - SurfNormAnalyticCostFunction(Eigen::Vector3d curr_point_, Eigen::Vector3d plane_unit_norm_, double negative_OA_dot_norm_); + SurfNormAnalyticCostFunction(Eigen::Vector3d curr_point, Eigen::Vector3d plane_unit_norm, double negative_OA_dot_norm); virtual ~SurfNormAnalyticCostFunction() {} virtual bool Evaluate(double const *const *parameters, double *residuals, double **jacobians) const; - Eigen::Vector3d curr_point; - Eigen::Vector3d plane_unit_norm; - double negative_OA_dot_norm; + Eigen::Vector3d curr_point_; + Eigen::Vector3d plane_unit_norm_; + double negative_OA_dot_norm_; }; -class PoseSE3Parameterization : public ceres::LocalParameterization { -public: - +class PoseSE3Parameterization : public ceres::LocalParameterization +{ + public: PoseSE3Parameterization() {} virtual ~PoseSE3Parameterization() {} virtual bool Plus(const double* x, const double* delta, double* x_plus_delta) const; @@ -46,8 +48,3 @@ class PoseSE3Parameterization : public ceres::LocalParameterization { virtual int GlobalSize() const { return 7; } virtual int LocalSize() const { return 6; } }; - - - -#endif // _LIDAR_OPTIMIZATION_ANALYTIC_H_ - diff --git a/include/odomEstimationClass.h b/include/floam/odom_estimation_class.hpp similarity index 71% rename from include/odomEstimationClass.h rename to include/floam/odom_estimation_class.hpp index 8d9998e..269f90a 100644 --- a/include/odomEstimationClass.h +++ b/include/floam/odom_estimation_class.hpp @@ -1,45 +1,30 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro -#ifndef _ODOM_ESTIMATION_CLASS_H_ -#define _ODOM_ESTIMATION_CLASS_H_ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang -//std lib -#include -#include -#include +#pragma once -//PCL -#include -#include -#include -#include -#include +// pcl header #include -#include -#include +#include #include -//ceres +// ceres #include #include -//eigen +// eigen #include #include -//LOCAL LIB -#include "lidar.h" -#include "lidarOptimization.h" -#include +// local header +#include "floam/lidar_optimization.hpp" -class OdomEstimationClass -{ - public: - OdomEstimationClass(); - - void init(lidar::Lidar lidar_param, double map_resolution); +class OdomEstimationClass +{ + public: + OdomEstimationClass() = default; + void init(double map_resolution); void initMapWithPoints(const pcl::PointCloud::Ptr& edge_in, const pcl::PointCloud::Ptr& surf_in); void updatePointsToMap(const pcl::PointCloud::Ptr& edge_in, const pcl::PointCloud::Ptr& surf_in); void getMap(pcl::PointCloud::Ptr& laserCloudMap); @@ -47,35 +32,33 @@ class OdomEstimationClass Eigen::Isometry3d odom; pcl::PointCloud::Ptr laserCloudCornerMap; pcl::PointCloud::Ptr laserCloudSurfMap; + private: - //optimization variable + // optimization variable double parameters[7] = {0, 0, 0, 1, 0, 0, 0}; Eigen::Map q_w_curr = Eigen::Map(parameters); Eigen::Map t_w_curr = Eigen::Map(parameters + 4); Eigen::Isometry3d last_odom; - //kd-tree + // kd-tree pcl::KdTreeFLANN::Ptr kdtreeEdgeMap; pcl::KdTreeFLANN::Ptr kdtreeSurfMap; - //points downsampling before add to map + // points downsampling before add to map pcl::VoxelGrid downSizeFilterEdge; pcl::VoxelGrid downSizeFilterSurf; - //local map + // local map pcl::CropBox cropBoxFilter; - //optimization count + // optimization count int optimization_count; - //function + // function void addEdgeCostFactor(const pcl::PointCloud::Ptr& pc_in, const pcl::PointCloud::Ptr& map_in, ceres::Problem& problem, ceres::LossFunction *loss_function); void addSurfCostFactor(const pcl::PointCloud::Ptr& pc_in, const pcl::PointCloud::Ptr& map_in, ceres::Problem& problem, ceres::LossFunction *loss_function); void addPointsToMap(const pcl::PointCloud::Ptr& downsampledEdgeCloud, const pcl::PointCloud::Ptr& downsampledSurfCloud); void pointAssociateToMap(pcl::PointXYZI const *const pi, pcl::PointXYZI *const po); void downSamplingToMap(const pcl::PointCloud::Ptr& edge_pc_in, pcl::PointCloud::Ptr& edge_pc_out, const pcl::PointCloud::Ptr& surf_pc_in, pcl::PointCloud::Ptr& surf_pc_out); }; - -#endif // _ODOM_ESTIMATION_CLASS_H_ - diff --git a/include/floam/odom_estimation_node.hpp b/include/floam/odom_estimation_node.hpp new file mode 100644 index 0000000..27cc7d2 --- /dev/null +++ b/include/floam/odom_estimation_node.hpp @@ -0,0 +1,47 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +#pragma once + +// ros header +#include +#include +#include +#include + +// c++ header +#include +#include + +// local header +#include "floam/lidar.hpp" +#include "floam/odom_estimation_class.hpp" + + +class OdomEstimationNode: public rclcpp::Node +{ + public: + OdomEstimationNode(); + void odom_estimation(); + + private: + OdomEstimationClass odomEstimation_; + std::queue pointCloudEdgeBuf_; + std::queue pointCloudSurfBuf_; + std::mutex mutex_lock_; + lidar::Lidar lidar_param_; + + rclcpp::Subscription::SharedPtr subEdgeLaserCloud_; + rclcpp::Subscription::SharedPtr subSurfLaserCloud_; + + rclcpp::Publisher::SharedPtr pubLaserOdometry_; + + std::unique_ptr br_; + + void velodyneSurfHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg); + void velodyneEdgeHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg); + + bool is_odom_inited_; + double total_time_; + int total_frame_; +}; diff --git a/include/laserMappingClass.h b/include/laserMappingClass.h deleted file mode 100644 index 7714446..0000000 --- a/include/laserMappingClass.h +++ /dev/null @@ -1,67 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro - -#ifndef _LASER_MAPPING_H_ -#define _LASER_MAPPING_H_ - -//PCL lib -#include -#include -#include -#include -#include -#include - -//eigen lib -#include -#include - -//c++ lib -#include -#include -#include - - -#define LASER_CELL_WIDTH 50.0 -#define LASER_CELL_HEIGHT 50.0 -#define LASER_CELL_DEPTH 50.0 - -//separate map as many sub point clouds - -#define LASER_CELL_RANGE_HORIZONTAL 2 -#define LASER_CELL_RANGE_VERTICAL 2 - - -class LaserMappingClass -{ - - public: - LaserMappingClass(); - void init(double map_resolution); - void updateCurrentPointsToMap(const pcl::PointCloud::Ptr& pc_in, const Eigen::Isometry3d& pose_current); - pcl::PointCloud::Ptr getMap(void); - - private: - int origin_in_map_x; - int origin_in_map_y; - int origin_in_map_z; - int map_width; - int map_height; - int map_depth; - std::vector::Ptr>>> map; - pcl::VoxelGrid downSizeFilter; - - void addWidthCellNegative(void); - void addWidthCellPositive(void); - void addHeightCellNegative(void); - void addHeightCellPositive(void); - void addDepthCellNegative(void); - void addDepthCellPositive(void); - void checkPoints(int& x, int& y, int& z); - -}; - - -#endif // _LASER_MAPPING_H_ - diff --git a/include/laserProcessingClass.h b/include/laserProcessingClass.h deleted file mode 100644 index 5dcedae..0000000 --- a/include/laserProcessingClass.h +++ /dev/null @@ -1,48 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro -#ifndef _LASER_PROCESSING_CLASS_H_ -#define _LASER_PROCESSING_CLASS_H_ - -#include -#include - -#include -#include -#include -#include -#include - -#include "lidar.h" - -//points covariance class -class Double2d{ -public: - int id; - double value; - Double2d(int id_in, double value_in); -}; -//points info class -class PointsInfo{ -public: - int layer; - double time; - PointsInfo(int layer_in, double time_in); -}; - - -class LaserProcessingClass -{ - public: - LaserProcessingClass(); - void init(lidar::Lidar lidar_param_in); - void featureExtraction(const pcl::PointCloud::Ptr& pc_in, pcl::PointCloud::Ptr& pc_out_edge, pcl::PointCloud::Ptr& pc_out_surf); - void featureExtractionFromSector(const pcl::PointCloud::Ptr& pc_in, std::vector& cloudCurvature, pcl::PointCloud::Ptr& pc_out_edge, pcl::PointCloud::Ptr& pc_out_surf); - private: - lidar::Lidar lidar_param; -}; - - - -#endif // _LASER_PROCESSING_CLASS_H_ - diff --git a/include/lidar.h b/include/lidar.h deleted file mode 100644 index 0a516da..0000000 --- a/include/lidar.h +++ /dev/null @@ -1,40 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro -#ifndef _LIDAR_H_ -#define _LIDAR_H_ - -//define lidar parameter - -namespace lidar{ - -class Lidar -{ - public: - Lidar(); - - void setScanPeriod(double scan_period_in); - void setLines(double num_lines_in); - void setVerticalAngle(double vertical_angle_in); - void setVerticalResolution(double vertical_angle_resolution_in); - //by default is 100. pls do not change - void setMaxDistance(double max_distance_in); - void setMinDistance(double min_distance_in); - - double max_distance; - double min_distance; - int num_lines; - double scan_period; - int points_per_line; - double horizontal_angle_resolution; - double horizontal_angle; - double vertical_angle_resolution; - double vertical_angle; -}; - - -} - - -#endif // _LIDAR_H_ - diff --git a/launch/floam.launch b/launch/floam.launch deleted file mode 100644 index 610ef97..0000000 --- a/launch/floam.launch +++ /dev/null @@ -1,50 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/launch/floam.launch.py b/launch/floam.launch.py new file mode 100644 index 0000000..719b807 --- /dev/null +++ b/launch/floam.launch.py @@ -0,0 +1,51 @@ +from launch import LaunchDescription +from launch_ros.actions import Node +from launch.actions import ExecuteProcess +from ament_index_python.packages import get_package_share_directory +from os.path import join + + +def generate_launch_description(): + params = join( + get_package_share_directory("floam"), "params", "floam_params.yaml" + ) + + laser_processing_node = Node( + package="floam", + executable="laser_processing_node", + name="laser_processing_node", + parameters=[params] + ) + + odom_estimation_node = Node( + package="floam", + executable="odom_estimation_node", + name="odom_estimation_node", + parameters=[params] + ) + + bag_exec = ExecuteProcess( + cmd=["ros2", "bag", "play", "-r", "0.85", "/data/rosbag/Kitti/2011_09_30_0018"] + ) + + tf_node = Node( + package="tf2_ros", + executable="static_transform_publisher", + name="world2map_tf", + arguments=['0', '0', '0', '0', '0', '0', "world", "map"] + ) + + rviz_node = Node( + package="rviz2", + executable="rviz2", + name="rviz2", + arguments=["-d", join(get_package_share_directory("floam"), "rviz/", "floam.rviz")] + ) + + return LaunchDescription([ + laser_processing_node, + odom_estimation_node, + bag_exec, + tf_node, + rviz_node + ]) diff --git a/launch/floam_mapping.launch b/launch/floam_mapping.launch deleted file mode 100644 index 55a253c..0000000 --- a/launch/floam_mapping.launch +++ /dev/null @@ -1,52 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/launch/floam_mapping.launch.py b/launch/floam_mapping.launch.py new file mode 100644 index 0000000..970de43 --- /dev/null +++ b/launch/floam_mapping.launch.py @@ -0,0 +1,59 @@ +from launch import LaunchDescription +from launch_ros.actions import Node +from launch.actions import ExecuteProcess +from ament_index_python.packages import get_package_share_directory +from os.path import join + + +def generate_launch_description(): + params = join( + get_package_share_directory("floam"), "params", "floam_params.yaml" + ) + + laser_processing_node = Node( + package="floam", + executable="laser_processing_node", + name="laser_processing_node", + parameters=[params] + ) + + odom_estimation_node = Node( + package="floam", + executable="odom_estimation_node", + name="odom_estimation_node", + parameters=[params] + ) + + laser_mapping_node = Node( + package="floam", + executable="laser_mapping_node", + name="laser_mapping_node", + parameters=[params] + ) + + bag_exec = ExecuteProcess( + cmd=["ros2", "bag", "play", "-r", "0.5", "/data/rosbag/Kitti/2011_09_30_0018"] + ) + + tf_node = Node( + package="tf2_ros", + executable="static_transform_publisher", + name="world2map_tf", + arguments=['0', '0', '0', '0', '0', '0', "world", "map"] + ) + + rviz_node = Node( + package="rviz2", + executable="rviz2", + name="rviz2", + arguments=["-d", join(get_package_share_directory("floam"), "rviz/", "floam_mapping.rviz")] + ) + + return LaunchDescription([ + laser_processing_node, + odom_estimation_node, + laser_mapping_node, + bag_exec, + tf_node, + rviz_node, + ]) diff --git a/launch/floam_velodyne.launch b/launch/floam_velodyne.launch deleted file mode 100644 index a707fba..0000000 --- a/launch/floam_velodyne.launch +++ /dev/null @@ -1,45 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/package.xml b/package.xml index f974c0d..2de5def 100644 --- a/package.xml +++ b/package.xml @@ -1,38 +1,29 @@ - + + floam 0.0.0 + TODO: Package description + yi-chen + TODO: License declaration - - This work is created based on some optimization on ALOAM by HKUST and LOAM by CMU - + ament_cmake + rclcpp + sensor_msgs + nav_msgs + geometry_msgs + tf2 + tf2_ros + pcl_conversions + PCL + Eigen3 - Han Wang + ros2launch - BSD - https://wanghan.pro - Han Wang - - catkin - geometry_msgs - nav_msgs - roscpp - rospy - std_msgs - rosbag - sensor_msgs - tf - eigen_conversions - geometry_msgs - nav_msgs - sensor_msgs - roscpp - rospy - std_msgs - rosbag - tf - eigen_conversions + ament_lint_auto + ament_lint_common + ament_cmake diff --git a/params/floam_params.yaml b/params/floam_params.yaml new file mode 100644 index 0000000..128846c --- /dev/null +++ b/params/floam_params.yaml @@ -0,0 +1,25 @@ +/laser_processing_node: + ros__parameters: + max_dist: 90.0 + min_dist: 3.0 + scan_line: 64 + scan_period: 0.1 + vertical_angle: 2.0 + +/odom_estimation_node: + ros__parameters: + map_resolution: 0.4 + max_dist: 90.0 + min_dist: 3.0 + scan_line: 64 + scan_period: 0.1 + vertical_angle: 2.0 + +/laser_mapping_node: + ros__parameters: + map_resolution: 0.4 + max_dist: 90.0 + min_dist: 3.0 + scan_line: 64 + scan_period: 0.1 + vertical_angle: 2.0 diff --git a/rviz/floam.rviz b/rviz/floam.rviz index a5a4af6..5435eb8 100644 --- a/rviz/floam.rviz +++ b/rviz/floam.rviz @@ -1,44 +1,31 @@ Panels: - - Class: rviz/Displays + - Class: rviz_common/Displays Help Height: 78 Name: Displays Property Tree Widget: Expanded: - - /Status1 - - /Grid1 - - /TF1 - /TF1/Frames1 Splitter Ratio: 0.5 - Tree Height: 530 - - Class: rviz/Selection + Tree Height: 565 + - Class: rviz_common/Selection Name: Selection - - Class: rviz/Tool Properties + - Class: rviz_common/Tool Properties Expanded: - - /2D Pose Estimate1 - - /2D Nav Goal1 + - /2D Goal Pose1 - /Publish Point1 Name: Tool Properties Splitter Ratio: 0.5886790156364441 - - Class: rviz/Views + - Class: rviz_common/Views Expanded: - /Current View1 Name: Views Splitter Ratio: 0.5 - - Class: rviz/Time - Experimental: false - Name: Time - SyncMode: 0 - SyncSource: Image -Preferences: - PromptSaveOnExit: true -Toolbars: - toolButtonStyle: 2 Visualization Manager: Class: "" Displays: - Alpha: 0.5 - Cell Size: 50 - Class: rviz/Grid + Cell Size: 1 + Class: rviz_default_plugins/Grid Color: 160; 160; 164 Enabled: true Line Style: @@ -51,22 +38,24 @@ Visualization Manager: Y: 0 Z: 0 Plane: XY - Plane Cell Count: 15 + Plane Cell Count: 10 Reference Frame: Value: true - - Class: rviz/Image + - Class: rviz_default_plugins/Image Enabled: true - Image Topic: /cam03/image_raw Max Value: 1 Median window: 5 Min Value: 0 Name: Image Normalize Range: true - Queue Size: 2 - Transport Hint: raw - Unreliable: false + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /cam03/image_raw Value: true - - Class: rviz/TF + - Class: rviz_default_plugins/TF Enabled: true Frame Timeout: 15 Frames: @@ -82,16 +71,16 @@ Visualization Manager: cam03: Value: false imu: - Value: false + Value: true map: - Value: false + Value: true velodyne: Value: true world: Value: true - Marker Scale: 100 + Marker Scale: 25 Name: TF - Show Arrows: false + Show Arrows: true Show Axes: true Show Names: true Tree: @@ -115,12 +104,12 @@ Visualization Manager: - Alpha: 1 Autocompute Intensity Bounds: true Autocompute Value Bounds: - Max Value: 4.832673072814941 - Min Value: -8.602354049682617 + Max Value: 10 + Min Value: -10 Value: true Axis: Z Channel Name: intensity - Class: rviz/PointCloud2 + Class: rviz_default_plugins/PointCloud2 Color: 255; 255; 255 Color Transformer: Intensity Decay Time: 0 @@ -132,122 +121,97 @@ Visualization Manager: Min Intensity: 0 Name: raw_data Position Transformer: XYZ - Queue Size: 10 Selectable: true Size (Pixels): 1 Size (m): 0.009999999776482582 Style: Points - Topic: /velodyne_points - Unreliable: false + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /velodyne_points Use Fixed Frame: true Use rainbow: true Value: true - - Alpha: 1 - Buffer Length: 1 - Class: rviz/Path - Color: 239; 41; 41 - Enabled: true - Head Diameter: 0.30000001192092896 - Head Length: 0.20000000298023224 - Length: 0.30000001192092896 - Line Style: Lines - Line Width: 0.029999999329447746 - Name: ground_truth - Offset: - X: 0 - Y: 0 - Z: 0 - Pose Color: 255; 85; 255 - Pose Style: None - Radius: 0.029999999329447746 - Shaft Diameter: 0.10000000149011612 - Shaft Length: 0.10000000149011612 - Topic: /gt/trajectory - Unreliable: false - Value: true - - Alpha: 1 - Buffer Length: 1 - Class: rviz/Path - Color: 0; 255; 0 - Enabled: true - Head Diameter: 0.30000001192092896 - Head Length: 0.20000000298023224 - Length: 0.30000001192092896 - Line Style: Lines - Line Width: 0.029999999329447746 - Name: floam_result - Offset: - X: 0 - Y: 0 - Z: 0 - Pose Color: 255; 85; 255 - Pose Style: None - Radius: 0.029999999329447746 - Shaft Diameter: 0.10000000149011612 - Shaft Length: 0.10000000149011612 - Topic: /base_link/trajectory - Unreliable: false - Value: true Enabled: true Global Options: - Background Color: 0; 0; 0 - Default Light: true + Background Color: 48; 48; 48 Fixed Frame: world Frame Rate: 30 Name: root Tools: - - Class: rviz/Interact + - Class: rviz_default_plugins/Interact Hide Inactive Objects: true - - Class: rviz/MoveCamera - - Class: rviz/Select - - Class: rviz/FocusCamera - - Class: rviz/Measure - - Class: rviz/SetInitialPose - Theta std deviation: 0.2617993950843811 - Topic: /initialpose - X std deviation: 0.5 - Y std deviation: 0.5 - - Class: rviz/SetGoal - Topic: /move_base_simple/goal - - Class: rviz/PublishPoint + - Class: rviz_default_plugins/MoveCamera + - Class: rviz_default_plugins/Select + - Class: rviz_default_plugins/FocusCamera + - Class: rviz_default_plugins/Measure + Line color: 128; 128; 0 + - Class: rviz_default_plugins/SetInitialPose + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /initialpose + - Class: rviz_default_plugins/SetGoal + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /goal_pose + - Class: rviz_default_plugins/PublishPoint Single click: true - Topic: /clicked_point + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /clicked_point + Transformation: + Current: + Class: rviz_default_plugins/TF Value: true Views: Current: - Angle: -1.5749943256378174 - Class: rviz/TopDownOrtho + Class: rviz_default_plugins/Orbit + Distance: 84.29740905761719 Enable Stereo Rendering: Stereo Eye Separation: 0.05999999865889549 Stereo Focal Distance: 1 Swap Stereo Eyes: false Value: false + Focal Point: + X: 0 + Y: 0 + Z: 0 + Focal Shape Fixed Size: false + Focal Shape Size: 0.05000000074505806 Invert Z Axis: false Name: Current View Near Clip Distance: 0.009999999776482582 - Scale: 2.7801454067230225 - Target Frame: body - Value: TopDownOrtho (rviz) - X: 84.71299743652344 - Y: -50.52915954589844 + Pitch: 0.9353980422019958 + Target Frame: base_link + Value: Orbit (rviz) + Yaw: 1.6353968381881714 Saved: ~ Window Geometry: Displays: collapsed: false - Height: 1025 + Height: 1056 Hide Left Dock: false Hide Right Dock: false Image: collapsed: false - QMainWindow State: 000000ff00000000fd00000004000000000000016a00000363fc020000000afb0000001200530065006c0065006300740069006f006e00000001e10000009b0000005c00fffffffb0000001e0054006f006f006c002000500072006f007000650072007400690065007302000001ed000001df00000185000000a3fb000000120056006900650077007300200054006f006f02000001df000002110000018500000122fb000000200054006f006f006c002000500072006f0070006500720074006900650073003203000002880000011d000002210000017afb000000100044006900730070006c006100790073010000003d0000029d000000c900fffffffb0000002000730065006c0065006300740069006f006e00200062007500660066006500720200000138000000aa0000023a00000294fb00000014005700690064006500530074006500720065006f02000000e6000000d2000003ee0000030bfb0000000c004b0069006e0065006300740200000186000001060000030c00000261fb0000000a0049006d00610067006501000002e0000000c00000001600fffffffb0000000600490053004300000002d9000000c70000000000000000000000010000010f00000363fc0200000003fb0000001e0054006f006f006c002000500072006f00700065007200740069006500730100000041000000780000000000000000fb0000000a00560069006500770073010000003d00000363000000a400fffffffb0000001200530065006c0065006300740069006f006e010000025a000000b200000000000000000000000200000490000000a9fc0100000001fb0000000a00560069006500770073030000004e00000080000002e100000197000000030000073d0000003efc0100000002fb0000000800540069006d006501000000000000073d000002eb00fffffffb0000000800540069006d00650100000000000004500000000000000000000004b80000036300000004000000040000000800000008fc0000000100000002000000010000000a0054006f006f006c00730100000000ffffffff0000000000000000 + QMainWindow State: 000000ff00000000fd0000000400000000000001a5000002c3fc0200000008fb0000001200530065006c0065006300740069006f006e00000001e10000009b0000005d00fffffffb0000001e0054006f006f006c002000500072006f007000650072007400690065007302000001ed000001df00000185000000a3fb000000120056006900650077007300200054006f006f02000001df000002110000018500000122fb000000200054006f006f006c002000500072006f0070006500720074006900650073003203000002880000011d000002210000017afb000000100044006900730070006c006100790073010000013e000002c3000000cc00fffffffb0000002000730065006c0065006300740069006f006e00200062007500660066006500720200000138000000aa0000023a00000294fb00000014005700690064006500530074006500720065006f02000000e6000000d2000003ee0000030bfb0000000c004b0069006e0065006300740200000186000001060000030c00000261000000010000010f000002c3fc0200000003fb0000001e0054006f006f006c002000500072006f00700065007200740069006500730100000041000000780000000000000000fb0000000a00560069006500770073010000013e000002c3000000a900fffffffb0000001200530065006c0065006300740069006f006e010000025a000000b20000000000000000000000020000078000000111fc0100000002fb0000000a0049006d0061006700650100000000000007800000006100fffffffb0000000a00560069006500770073030000004e00000080000002e10000019700000003000004420000003efc0100000002fb0000000800540069006d00650100000000000004420000000000000000fb0000000800540069006d00650100000000000004500000000000000000000004c0000002c300000004000000040000000800000008fc0000000100000002000000010000000a0054006f006f006c00730100000000ffffffff0000000000000000 Selection: collapsed: false - Time: - collapsed: false Tool Properties: collapsed: false Views: collapsed: false - Width: 1853 - X: 795 - Y: 27 + Width: 1920 + X: 0 + Y: 24 diff --git a/rviz/floam_mapping.rviz b/rviz/floam_mapping.rviz index 028e2a7..2578127 100644 --- a/rviz/floam_mapping.rviz +++ b/rviz/floam_mapping.rviz @@ -1,45 +1,33 @@ Panels: - - Class: rviz/Displays + - Class: rviz_common/Displays Help Height: 78 Name: Displays Property Tree Widget: Expanded: - - /Global Options1 - - /TF1/Frames1 - - /map1 + - /TF1 Splitter Ratio: 0.5 - Tree Height: 530 - - Class: rviz/Selection + Tree Height: 616 + - Class: rviz_common/Selection Name: Selection - - Class: rviz/Tool Properties + - Class: rviz_common/Tool Properties Expanded: - - /2D Pose Estimate1 - - /2D Nav Goal1 + - /2D Goal Pose1 - /Publish Point1 Name: Tool Properties Splitter Ratio: 0.5886790156364441 - - Class: rviz/Views + - Class: rviz_common/Views Expanded: - /Current View1 Name: Views Splitter Ratio: 0.5 - - Class: rviz/Time - Experimental: false - Name: Time - SyncMode: 0 - SyncSource: Image -Preferences: - PromptSaveOnExit: true -Toolbars: - toolButtonStyle: 2 Visualization Manager: Class: "" Displays: - Alpha: 0.5 - Cell Size: 50 - Class: rviz/Grid + Cell Size: 1 + Class: rviz_default_plugins/Grid Color: 160; 160; 164 - Enabled: false + Enabled: true Line Style: Line Width: 0.029999999329447746 Value: Lines @@ -50,22 +38,24 @@ Visualization Manager: Y: 0 Z: 0 Plane: XY - Plane Cell Count: 15 + Plane Cell Count: 10 Reference Frame: - Value: false - - Class: rviz/Image + Value: true + - Class: rviz_default_plugins/Image Enabled: true - Image Topic: /cam03/image_raw Max Value: 1 Median window: 5 Min Value: 0 Name: Image Normalize Range: true - Queue Size: 2 - Transport Hint: raw - Unreliable: false + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /cam03/image_raw Value: true - - Class: rviz/TF + - Class: rviz_default_plugins/TF Enabled: true Frame Timeout: 15 Frames: @@ -83,14 +73,14 @@ Visualization Manager: imu: Value: false map: - Value: false + Value: true velodyne: - Value: false + Value: true world: Value: true - Marker Scale: 50 + Marker Scale: 25 Name: TF - Show Arrows: false + Show Arrows: true Show Axes: true Show Names: true Tree: @@ -114,16 +104,16 @@ Visualization Manager: - Alpha: 1 Autocompute Intensity Bounds: true Autocompute Value Bounds: - Max Value: 0.21922564506530762 - Min Value: -11.99945068359375 + Max Value: 10 + Min Value: -10 Value: true Axis: Z Channel Name: intensity - Class: rviz/PointCloud2 + Class: rviz_default_plugins/PointCloud2 Color: 255; 255; 255 Color Transformer: Intensity Decay Time: 0 - Enabled: false + Enabled: true Invert Rainbow: false Max Color: 255; 255; 255 Max Intensity: 0.9900000095367432 @@ -131,25 +121,28 @@ Visualization Manager: Min Intensity: 0 Name: raw_data Position Transformer: XYZ - Queue Size: 10 Selectable: true - Size (Pixels): 2 + Size (Pixels): 1 Size (m): 0.009999999776482582 Style: Points - Topic: /velodyne_points - Unreliable: false + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /velodyne_points Use Fixed Frame: true Use rainbow: true - Value: false - - Alpha: 0.699999988079071 + Value: true + - Alpha: 1 Autocompute Intensity Bounds: true Autocompute Value Bounds: - Max Value: 8.244071960449219 - Min Value: -20.63126564025879 + Max Value: 10 + Min Value: -10 Value: true Axis: Z Channel Name: intensity - Class: rviz/PointCloud2 + Class: rviz_default_plugins/PointCloud2 Color: 255; 255; 255 Color Transformer: Intensity Decay Time: 0 @@ -161,127 +154,92 @@ Visualization Manager: Min Intensity: 0 Name: map Position Transformer: XYZ - Queue Size: 10 Selectable: true - Size (Pixels): 1 - Size (m): 0.3499999940395355 - Style: Squares - Topic: /map - Unreliable: false + Size (Pixels): 3 + Size (m): 0.009999999776482582 + Style: Points + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /map Use Fixed Frame: true Use rainbow: true Value: true - - Alpha: 1 - Buffer Length: 1 - Class: rviz/Path - Color: 239; 41; 41 - Enabled: true - Head Diameter: 0.30000001192092896 - Head Length: 0.20000000298023224 - Length: 0.30000001192092896 - Line Style: Lines - Line Width: 0.029999999329447746 - Name: ground_truth - Offset: - X: 0 - Y: 0 - Z: 0 - Pose Color: 255; 85; 255 - Pose Style: None - Radius: 0.029999999329447746 - Shaft Diameter: 0.10000000149011612 - Shaft Length: 0.10000000149011612 - Topic: /gt/trajectory - Unreliable: false - Value: true - - Alpha: 1 - Buffer Length: 1 - Class: rviz/Path - Color: 25; 255; 0 - Enabled: true - Head Diameter: 0.30000001192092896 - Head Length: 0.20000000298023224 - Length: 0.30000001192092896 - Line Style: Lines - Line Width: 0.029999999329447746 - Name: floam_result - Offset: - X: 0 - Y: 0 - Z: 0 - Pose Color: 255; 85; 255 - Pose Style: None - Radius: 0.029999999329447746 - Shaft Diameter: 0.10000000149011612 - Shaft Length: 0.10000000149011612 - Topic: /base_link/trajectory - Unreliable: false - Value: true Enabled: true Global Options: - Background Color: 136; 138; 133 - Default Light: true - Fixed Frame: base_link + Background Color: 48; 48; 48 + Fixed Frame: world Frame Rate: 30 Name: root Tools: - - Class: rviz/Interact + - Class: rviz_default_plugins/Interact Hide Inactive Objects: true - - Class: rviz/MoveCamera - - Class: rviz/Select - - Class: rviz/FocusCamera - - Class: rviz/Measure - - Class: rviz/SetInitialPose - Theta std deviation: 0.2617993950843811 - Topic: /initialpose - X std deviation: 0.5 - Y std deviation: 0.5 - - Class: rviz/SetGoal - Topic: /move_base_simple/goal - - Class: rviz/PublishPoint + - Class: rviz_default_plugins/MoveCamera + - Class: rviz_default_plugins/Select + - Class: rviz_default_plugins/FocusCamera + - Class: rviz_default_plugins/Measure + Line color: 128; 128; 0 + - Class: rviz_default_plugins/SetInitialPose + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /initialpose + - Class: rviz_default_plugins/SetGoal + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /goal_pose + - Class: rviz_default_plugins/PublishPoint Single click: true - Topic: /clicked_point + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /clicked_point + Transformation: + Current: + Class: rviz_default_plugins/TF Value: true Views: Current: - Class: rviz/ThirdPersonFollower - Distance: 86.67870330810547 + Angle: 0 + Class: rviz_default_plugins/TopDownOrtho Enable Stereo Rendering: Stereo Eye Separation: 0.05999999865889549 Stereo Focal Distance: 1 Swap Stereo Eyes: false Value: false - Focal Point: - X: -35.922298431396484 - Y: 0.5339305400848389 - Z: 6.388764450093731e-5 - Focal Shape Fixed Size: true - Focal Shape Size: 0.05000000074505806 Invert Z Axis: false Name: Current View Near Clip Distance: 0.009999999776482582 - Pitch: 0.9953982830047607 - Target Frame: body - Value: ThirdPersonFollower (rviz) - Yaw: 3.2104086875915527 + Scale: 6.787409782409668 + Target Frame: base_link + Value: TopDownOrtho (rviz_default_plugins) + X: -7.8558454513549805 + Y: -1.1727551221847534 Saved: ~ Window Geometry: Displays: collapsed: false - Height: 1025 + Height: 1056 Hide Left Dock: false Hide Right Dock: false Image: collapsed: false - QMainWindow State: 000000ff00000000fd00000004000000000000016a00000363fc020000000afb0000001200530065006c0065006300740069006f006e00000001e10000009b0000005c00fffffffb0000001e0054006f006f006c002000500072006f007000650072007400690065007302000001ed000001df00000185000000a3fb000000120056006900650077007300200054006f006f02000001df000002110000018500000122fb000000200054006f006f006c002000500072006f0070006500720074006900650073003203000002880000011d000002210000017afb000000100044006900730070006c006100790073010000003d0000029d000000c900fffffffb0000002000730065006c0065006300740069006f006e00200062007500660066006500720200000138000000aa0000023a00000294fb00000014005700690064006500530074006500720065006f02000000e6000000d2000003ee0000030bfb0000000c004b0069006e0065006300740200000186000001060000030c00000261fb0000000a0049006d00610067006501000002e0000000c00000001600fffffffb0000000600490053004300000002d9000000c70000000000000000000000010000010f00000363fc0200000003fb0000001e0054006f006f006c002000500072006f00700065007200740069006500730100000041000000780000000000000000fb0000000a00560069006500770073010000003d00000363000000a400fffffffb0000001200530065006c0065006300740069006f006e010000025a000000b200000000000000000000000200000490000000a9fc0100000001fb0000000a00560069006500770073030000004e00000080000002e100000197000000030000073d0000003efc0100000002fb0000000800540069006d006501000000000000073d000002eb00fffffffb0000000800540069006d00650100000000000004500000000000000000000004b80000036300000004000000040000000800000008fc0000000100000002000000010000000a0054006f006f006c00730100000000ffffffff0000000000000000 + QMainWindow State: 000000ff00000000fd0000000400000000000001a5000002f6fc0200000008fb0000001200530065006c0065006300740069006f006e00000001e10000009b0000005d00fffffffb0000001e0054006f006f006c002000500072006f007000650072007400690065007302000001ed000001df00000185000000a3fb000000120056006900650077007300200054006f006f02000001df000002110000018500000122fb000000200054006f006f006c002000500072006f0070006500720074006900650073003203000002880000011d000002210000017afb000000100044006900730070006c006100790073010000010b000002f6000000cc00fffffffb0000002000730065006c0065006300740069006f006e00200062007500660066006500720200000138000000aa0000023a00000294fb00000014005700690064006500530074006500720065006f02000000e6000000d2000003ee0000030bfb0000000c004b0069006e0065006300740200000186000001060000030c00000261000000010000010f000002f6fc0200000003fb0000001e0054006f006f006c002000500072006f00700065007200740069006500730100000041000000780000000000000000fb0000000a00560069006500770073010000010b000002f6000000a900fffffffb0000001200530065006c0065006300740069006f006e010000025a000000b200000000000000000000000200000780000000defc0100000002fb0000000a0049006d0061006700650100000000000007800000006100fffffffb0000000a00560069006500770073030000004e00000080000002e10000019700000003000004420000003efc0100000002fb0000000800540069006d00650100000000000004420000000000000000fb0000000800540069006d00650100000000000004500000000000000000000004c0000002f600000004000000040000000800000008fc0000000100000002000000010000000a0054006f006f006c00730100000000ffffffff0000000000000000 Selection: collapsed: false - Time: - collapsed: false Tool Properties: collapsed: false Views: collapsed: false - Width: 1853 - X: 795 - Y: 27 + Width: 1920 + X: 0 + Y: 24 diff --git a/rviz/floam_octomap.rviz b/rviz/floam_octomap.rviz new file mode 100644 index 0000000..3554f47 --- /dev/null +++ b/rviz/floam_octomap.rviz @@ -0,0 +1,233 @@ +Panels: + - Class: rviz_common/Displays + Help Height: 78 + Name: Displays + Property Tree Widget: + Expanded: + - /Global Options1 + - /raw_data1 + - /map1 + - /map1/Status1 + - /map1/Namespaces1 + Splitter Ratio: 0.5 + Tree Height: 565 + - Class: rviz_common/Selection + Name: Selection + - Class: rviz_common/Tool Properties + Expanded: + - /2D Goal Pose1 + - /Publish Point1 + Name: Tool Properties + Splitter Ratio: 0.5886790156364441 + - Class: rviz_common/Views + Expanded: + - /Current View1 + Name: Views + Splitter Ratio: 0.5 +Visualization Manager: + Class: "" + Displays: + - Alpha: 0.5 + Cell Size: 1 + Class: rviz_default_plugins/Grid + Color: 160; 160; 164 + Enabled: true + Line Style: + Line Width: 0.029999999329447746 + Value: Lines + Name: Grid + Normal Cell Count: 0 + Offset: + X: 0 + Y: 0 + Z: 0 + Plane: XY + Plane Cell Count: 10 + Reference Frame: + Value: true + - Class: rviz_default_plugins/Image + Enabled: true + Max Value: 1 + Median window: 5 + Min Value: 0 + Name: Image + Normalize Range: true + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /cam03/image_raw + Value: true + - Class: rviz_default_plugins/TF + Enabled: true + Frame Timeout: 15 + Frames: + All Enabled: true + base_link: + Value: true + cam00: + Value: true + cam01: + Value: true + cam02: + Value: true + cam03: + Value: true + imu: + Value: true + map: + Value: true + velodyne: + Value: true + world: + Value: true + Marker Scale: 15 + Name: TF + Show Arrows: true + Show Axes: true + Show Names: false + Tree: + world: + imu: + cam00: + {} + cam01: + {} + cam02: + {} + cam03: + {} + velodyne: + {} + map: + base_link: + {} + Update Interval: 0 + Value: true + - Alpha: 1 + Autocompute Intensity Bounds: true + Autocompute Value Bounds: + Max Value: 10 + Min Value: -10 + Value: true + Axis: Z + Channel Name: intensity + Class: rviz_default_plugins/PointCloud2 + Color: 255; 255; 255 + Color Transformer: Intensity + Decay Time: 0 + Enabled: true + Invert Rainbow: false + Max Color: 255; 255; 255 + Max Intensity: 0.9900000095367432 + Min Color: 0; 0; 0 + Min Intensity: 0 + Name: raw_data + Position Transformer: XYZ + Selectable: true + Size (Pixels): 1 + Size (m): 0.009999999776482582 + Style: Points + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /velodyne_points + Use Fixed Frame: true + Use rainbow: true + Value: true + - Class: rviz_default_plugins/MarkerArray + Enabled: true + Name: map + Namespaces: + map: true + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /occupied_cells_vis_array + Value: true + Enabled: true + Global Options: + Background Color: 48; 48; 48 + Fixed Frame: world + Frame Rate: 30 + Name: root + Tools: + - Class: rviz_default_plugins/Interact + Hide Inactive Objects: true + - Class: rviz_default_plugins/MoveCamera + - Class: rviz_default_plugins/Select + - Class: rviz_default_plugins/FocusCamera + - Class: rviz_default_plugins/Measure + Line color: 128; 128; 0 + - Class: rviz_default_plugins/SetInitialPose + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /initialpose + - Class: rviz_default_plugins/SetGoal + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /goal_pose + - Class: rviz_default_plugins/PublishPoint + Single click: true + Topic: + Depth: 5 + Durability Policy: Volatile + History Policy: Keep Last + Reliability Policy: Reliable + Value: /clicked_point + Transformation: + Current: + Class: rviz_default_plugins/TF + Value: true + Views: + Current: + Class: rviz_default_plugins/Orbit + Distance: 75.79230499267578 + Enable Stereo Rendering: + Stereo Eye Separation: 0.05999999865889549 + Stereo Focal Distance: 1 + Swap Stereo Eyes: false + Value: false + Focal Point: + X: 0 + Y: 0 + Z: 0 + Focal Shape Fixed Size: false + Focal Shape Size: 0.05000000074505806 + Invert Z Axis: false + Name: Current View + Near Clip Distance: 0.009999999776482582 + Pitch: 0.8903980851173401 + Target Frame: base_link + Value: Orbit (rviz) + Yaw: 0.7203983664512634 + Saved: ~ +Window Geometry: + Displays: + collapsed: false + Height: 1056 + Hide Left Dock: false + Hide Right Dock: false + Image: + collapsed: false + QMainWindow State: 000000ff00000000fd0000000400000000000001a5000002c3fc0200000008fb0000001200530065006c0065006300740069006f006e00000001e10000009b0000005d00fffffffb0000001e0054006f006f006c002000500072006f007000650072007400690065007302000001ed000001df00000185000000a3fb000000120056006900650077007300200054006f006f02000001df000002110000018500000122fb000000200054006f006f006c002000500072006f0070006500720074006900650073003203000002880000011d000002210000017afb000000100044006900730070006c006100790073010000013e000002c3000000cc00fffffffb0000002000730065006c0065006300740069006f006e00200062007500660066006500720200000138000000aa0000023a00000294fb00000014005700690064006500530074006500720065006f02000000e6000000d2000003ee0000030bfb0000000c004b0069006e0065006300740200000186000001060000030c00000261000000010000010f000002c3fc0200000003fb0000001e0054006f006f006c002000500072006f00700065007200740069006500730100000041000000780000000000000000fb0000000a00560069006500770073010000013e000002c3000000a900fffffffb0000001200530065006c0065006300740069006f006e010000025a000000b20000000000000000000000020000078000000111fc0100000002fb0000000a0049006d0061006700650100000000000007800000006100fffffffb0000000a00560069006500770073030000004e00000080000002e10000019700000003000004420000003efc0100000002fb0000000800540069006d00650100000000000004420000000000000000fb0000000800540069006d00650100000000000004500000000000000000000004c0000002c300000004000000040000000800000008fc0000000100000002000000010000000a0054006f006f006c00730100000000ffffffff0000000000000000 + Selection: + collapsed: false + Tool Properties: + collapsed: false + Views: + collapsed: false + Width: 1920 + X: 0 + Y: 24 diff --git a/rviz/floam_velodyne.rviz b/rviz/floam_velodyne.rviz deleted file mode 100644 index bfde8bc..0000000 --- a/rviz/floam_velodyne.rviz +++ /dev/null @@ -1,233 +0,0 @@ -Panels: - - Class: rviz/Displays - Help Height: 78 - Name: Displays - Property Tree Widget: - Expanded: - - /Grid1 - - /TF1 - - /TF1/Frames1 - - /map1 - - /map1/Autocompute Value Bounds1 - Splitter Ratio: 0.5 - Tree Height: 728 - - Class: rviz/Selection - Name: Selection - - Class: rviz/Tool Properties - Expanded: - - /2D Pose Estimate1 - - /2D Nav Goal1 - - /Publish Point1 - Name: Tool Properties - Splitter Ratio: 0.5886790156364441 - - Class: rviz/Views - Expanded: - - /Current View1 - Name: Views - Splitter Ratio: 0.5 - - Class: rviz/Time - Experimental: false - Name: Time - SyncMode: 0 - SyncSource: map -Preferences: - PromptSaveOnExit: true -Toolbars: - toolButtonStyle: 2 -Visualization Manager: - Class: "" - Displays: - - Alpha: 0.5 - Cell Size: 5 - Class: rviz/Grid - Color: 160; 160; 164 - Enabled: true - Line Style: - Line Width: 0.029999999329447746 - Value: Lines - Name: Grid - Normal Cell Count: 0 - Offset: - X: 0 - Y: 0 - Z: 0 - Plane: XY - Plane Cell Count: 15 - Reference Frame: - Value: true - - Class: rviz/TF - Enabled: true - Frame Timeout: 15 - Frames: - All Enabled: false - aft_mapped: - Value: false - base_link: - Value: true - map: - Value: true - world: - Value: false - Marker Scale: 3 - Name: TF - Show Arrows: false - Show Axes: true - Show Names: true - Tree: - world: - map: - aft_mapped: - {} - base_link: - {} - Update Interval: 0 - Value: true - - Alpha: 1 - Autocompute Intensity Bounds: true - Autocompute Value Bounds: - Max Value: 0.21922564506530762 - Min Value: -11.99945068359375 - Value: true - Axis: Z - Channel Name: intensity - Class: rviz/PointCloud2 - Color: 255; 255; 255 - Color Transformer: Intensity - Decay Time: 0 - Enabled: false - Invert Rainbow: false - Max Color: 255; 255; 255 - Max Intensity: 255 - Min Color: 0; 0; 0 - Min Intensity: 0 - Name: raw_data - Position Transformer: XYZ - Queue Size: 10 - Selectable: true - Size (Pixels): 1 - Size (m): 0.009999999776482582 - Style: Points - Topic: /velodyne_points_filtered - Unreliable: false - Use Fixed Frame: true - Use rainbow: true - Value: false - - Alpha: 0.699999988079071 - Autocompute Intensity Bounds: true - Autocompute Value Bounds: - Max Value: 2.685699939727783 - Min Value: -1.4920799732208252 - Value: false - Axis: Z - Channel Name: intensity - Class: rviz/PointCloud2 - Color: 255; 255; 255 - Color Transformer: AxisColor - Decay Time: 0 - Enabled: true - Invert Rainbow: false - Max Color: 255; 255; 255 - Max Intensity: 0.9565645456314087 - Min Color: 0; 0; 0 - Min Intensity: 0 - Name: map - Position Transformer: XYZ - Queue Size: 10 - Selectable: true - Size (Pixels): 2 - Size (m): 0.10000000149011612 - Style: Flat Squares - Topic: /map - Unreliable: false - Use Fixed Frame: true - Use rainbow: true - Value: true - - Alpha: 1 - Buffer Length: 1 - Class: rviz/Path - Color: 25; 255; 0 - Enabled: true - Head Diameter: 0.30000001192092896 - Head Length: 0.20000000298023224 - Length: 0.30000001192092896 - Line Style: Lines - Line Width: 0.029999999329447746 - Name: floam_result - Offset: - X: 0 - Y: 0 - Z: 0 - Pose Color: 255; 85; 255 - Pose Style: None - Radius: 0.029999999329447746 - Shaft Diameter: 0.10000000149011612 - Shaft Length: 0.10000000149011612 - Topic: /base_link/trajectory - Unreliable: false - Value: true - Enabled: true - Global Options: - Background Color: 0; 0; 0 - Default Light: true - Fixed Frame: map - Frame Rate: 30 - Name: root - Tools: - - Class: rviz/Interact - Hide Inactive Objects: true - - Class: rviz/MoveCamera - - Class: rviz/Select - - Class: rviz/FocusCamera - - Class: rviz/Measure - - Class: rviz/SetInitialPose - Theta std deviation: 0.2617993950843811 - Topic: /initialpose - X std deviation: 0.5 - Y std deviation: 0.5 - - Class: rviz/SetGoal - Topic: /move_base_simple/goal - - Class: rviz/PublishPoint - Single click: true - Topic: /clicked_point - Value: true - Views: - Current: - Class: rviz/ThirdPersonFollower - Distance: 16.892133712768555 - Enable Stereo Rendering: - Stereo Eye Separation: 0.05999999865889549 - Stereo Focal Distance: 1 - Swap Stereo Eyes: false - Value: false - Focal Point: - X: -1.854732632637024 - Y: -1.3892109394073486 - Z: 6.691127782687545e-5 - Focal Shape Fixed Size: true - Focal Shape Size: 0.05000000074505806 - Invert Z Axis: false - Name: Current View - Near Clip Distance: 0.009999999776482582 - Pitch: 0.924797773361206 - Target Frame: body - Value: ThirdPersonFollower (rviz) - Yaw: 1.1854360103607178 - Saved: ~ -Window Geometry: - Displays: - collapsed: false - Height: 1025 - Hide Left Dock: false - Hide Right Dock: false - QMainWindow State: 000000ff00000000fd00000004000000000000016a00000363fc020000000afb0000001200530065006c0065006300740069006f006e00000001e10000009b0000005c00fffffffb0000001e0054006f006f006c002000500072006f007000650072007400690065007302000001ed000001df00000185000000a3fb000000120056006900650077007300200054006f006f02000001df000002110000018500000122fb000000200054006f006f006c002000500072006f0070006500720074006900650073003203000002880000011d000002210000017afb000000100044006900730070006c006100790073010000003d00000363000000c900fffffffb0000002000730065006c0065006300740069006f006e00200062007500660066006500720200000138000000aa0000023a00000294fb00000014005700690064006500530074006500720065006f02000000e6000000d2000003ee0000030bfb0000000c004b0069006e0065006300740200000186000001060000030c00000261fb0000000a0049006d00610067006501000002e0000000c00000000000000000fb0000000600490053004300000002d9000000c70000000000000000000000010000010f00000363fc0200000003fb0000001e0054006f006f006c002000500072006f00700065007200740069006500730100000041000000780000000000000000fb0000000a00560069006500770073010000003d00000363000000a400fffffffb0000001200530065006c0065006300740069006f006e010000025a000000b200000000000000000000000200000490000000a9fc0100000001fb0000000a00560069006500770073030000004e00000080000002e100000197000000030000073d0000003efc0100000002fb0000000800540069006d006501000000000000073d000002eb00fffffffb0000000800540069006d00650100000000000004500000000000000000000004b80000036300000004000000040000000800000008fc0000000100000002000000010000000a0054006f006f006c00730100000000ffffffff0000000000000000 - Selection: - collapsed: false - Time: - collapsed: false - Tool Properties: - collapsed: false - Views: - collapsed: false - Width: 1853 - X: 795 - Y: 27 diff --git a/src/laserMappingClass.cpp b/src/laserMappingClass.cpp deleted file mode 100644 index f9db07d..0000000 --- a/src/laserMappingClass.cpp +++ /dev/null @@ -1,205 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro - -#include "laserMappingClass.h" - -void LaserMappingClass::init(double map_resolution){ - //init map - //init can have real object, but future added block does not need - for(int i=0;i::Ptr>> map_height_temp; - for(int j=0;j::Ptr> map_depth_temp; - for(int k=0;k::Ptr point_cloud_temp(new pcl::PointCloud); - map_depth_temp.push_back(point_cloud_temp); - } - map_height_temp.push_back(map_depth_temp); - } - map.push_back(map_height_temp); - } - - origin_in_map_x = LASER_CELL_RANGE_HORIZONTAL; - origin_in_map_y = LASER_CELL_RANGE_HORIZONTAL; - origin_in_map_z = LASER_CELL_RANGE_VERTICAL; - map_width = LASER_CELL_RANGE_HORIZONTAL*2+1; - map_height = LASER_CELL_RANGE_HORIZONTAL*2+1; - map_depth = LASER_CELL_RANGE_HORIZONTAL*2+1; - - //downsampling size - downSizeFilter.setLeafSize(map_resolution, map_resolution, map_resolution); -} - -void LaserMappingClass::addWidthCellNegative(void){ - std::vector::Ptr>> map_height_temp; - for(int j=0; j < map_height;j++){ - std::vector::Ptr> map_depth_temp; - for(int k=0;k< map_depth;k++){ - pcl::PointCloud::Ptr point_cloud_temp; - map_depth_temp.push_back(point_cloud_temp); - } - map_height_temp.push_back(map_depth_temp); - } - map.insert(map.begin(), map_height_temp); - - origin_in_map_x++; - map_width++; -} -void LaserMappingClass::addWidthCellPositive(void){ - std::vector::Ptr>> map_height_temp; - for(int j=0; j < map_height;j++){ - std::vector::Ptr> map_depth_temp; - for(int k=0;k< map_depth;k++){ - pcl::PointCloud::Ptr point_cloud_temp; - map_depth_temp.push_back(point_cloud_temp); - } - map_height_temp.push_back(map_depth_temp); - } - map.push_back(map_height_temp); - map_width++; -} -void LaserMappingClass::addHeightCellNegative(void){ - for(int i=0; i < map_width;i++){ - std::vector::Ptr> map_depth_temp; - for(int k=0;k::Ptr point_cloud_temp; - map_depth_temp.push_back(point_cloud_temp); - } - map[i].insert(map[i].begin(), map_depth_temp); - } - origin_in_map_y++; - map_height++; -} -void LaserMappingClass::addHeightCellPositive(void){ - for(int i=0; i < map_width;i++){ - std::vector::Ptr> map_depth_temp; - for(int k=0;k::Ptr point_cloud_temp; - map_depth_temp.push_back(point_cloud_temp); - } - map[i].push_back(map_depth_temp); - } - map_height++; -} -void LaserMappingClass::addDepthCellNegative(void){ - for(int i=0; i < map_width;i++){ - for(int j=0;j< map_height;j++){ - pcl::PointCloud::Ptr point_cloud_temp; - map[i][j].insert(map[i][j].begin(), point_cloud_temp); - } - } - origin_in_map_z++; - map_depth++; -} -void LaserMappingClass::addDepthCellPositive(void){ - for(int i=0; i < map_width;i++){ - for(int j=0;j< map_height;j++){ - pcl::PointCloud::Ptr point_cloud_temp; - map[i][j].push_back(point_cloud_temp); - } - } - map_depth++; -} - -//extend map is points exceed size -void LaserMappingClass::checkPoints(int& x, int& y, int& z){ - - while(x + LASER_CELL_RANGE_HORIZONTAL> map_width-1){ - addWidthCellPositive(); - } - while(x-LASER_CELL_RANGE_HORIZONTAL<0){ - addWidthCellNegative(); - x++; - } - while(y + LASER_CELL_RANGE_HORIZONTAL> map_height-1){ - addHeightCellPositive(); - } - while(y-LASER_CELL_RANGE_HORIZONTAL<0){ - addHeightCellNegative(); - y++; - } - while(z + LASER_CELL_RANGE_VERTICAL> map_depth-1){ - addDepthCellPositive(); - } - while(z-LASER_CELL_RANGE_VERTICAL<0){ - addDepthCellNegative(); - z++; - } - - //initialize - //create object if area is null - for(int i=x-LASER_CELL_RANGE_HORIZONTAL;i::Ptr point_cloud_temp(new pcl::PointCloud()); - map[i][j][k] = point_cloud_temp; - } - - } - - } - - } -} - -//update points to map -void LaserMappingClass::updateCurrentPointsToMap(const pcl::PointCloud::Ptr& pc_in, const Eigen::Isometry3d& pose_current){ - - int currentPosIdX = int(std::floor(pose_current.translation().x() / LASER_CELL_WIDTH + 0.5)) + origin_in_map_x; - int currentPosIdY = int(std::floor(pose_current.translation().y() / LASER_CELL_HEIGHT + 0.5)) + origin_in_map_y; - int currentPosIdZ = int(std::floor(pose_current.translation().z() / LASER_CELL_DEPTH + 0.5)) + origin_in_map_z; - - //check is submap is null - checkPoints(currentPosIdX,currentPosIdY,currentPosIdZ); - - pcl::PointCloud::Ptr transformed_pc(new pcl::PointCloud()); - pcl::transformPointCloud(*pc_in, *transformed_pc, pose_current.cast()); - - //save points - for (int i = 0; i < (int)transformed_pc->points.size(); i++) - { - pcl::PointXYZI point_temp = transformed_pc->points[i]; - //for visualization only - point_temp.intensity = std::min(1.0 , std::max(pc_in->points[i].z+2.0, 0.0) / 5); - int currentPointIdX = int(std::floor(point_temp.x / LASER_CELL_WIDTH + 0.5)) + origin_in_map_x; - int currentPointIdY = int(std::floor(point_temp.y / LASER_CELL_HEIGHT + 0.5)) + origin_in_map_y; - int currentPointIdZ = int(std::floor(point_temp.z / LASER_CELL_DEPTH + 0.5)) + origin_in_map_z; - - map[currentPointIdX][currentPointIdY][currentPointIdZ]->push_back(point_temp); - - } - - //filtering points - for(int i=currentPosIdX-LASER_CELL_RANGE_HORIZONTAL;i::Ptr LaserMappingClass::getMap(void){ - pcl::PointCloud::Ptr laserCloudMap = pcl::PointCloud::Ptr(new pcl::PointCloud()); - for (int i = 0; i < map_width; i++){ - for (int j = 0; j < map_height; j++){ - for (int k = 0; k < map_depth; k++){ - if(map[i][j][k]!=NULL){ - *laserCloudMap += *(map[i][j][k]); - } - } - } - } - return laserCloudMap; -} - -LaserMappingClass::LaserMappingClass(){ - -} - diff --git a/src/laserMappingNode.cpp b/src/laserMappingNode.cpp deleted file mode 100644 index 4f051bf..0000000 --- a/src/laserMappingNode.cpp +++ /dev/null @@ -1,137 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro - -//c++ lib -#include -#include -#include -#include -#include -#include - -//ros lib -#include -#include -#include -#include -#include - -//pcl lib -#include -#include -#include - -//local lib -#include "laserMappingClass.h" -#include "lidar.h" - - -LaserMappingClass laserMapping; -lidar::Lidar lidar_param; -std::mutex mutex_lock; -std::queue odometryBuf; -std::queue pointCloudBuf; - -ros::Publisher map_pub; -void odomCallback(const nav_msgs::Odometry::ConstPtr &msg) -{ - mutex_lock.lock(); - odometryBuf.push(msg); - mutex_lock.unlock(); -} - -void velodyneHandler(const sensor_msgs::PointCloud2ConstPtr &laserCloudMsg) -{ - mutex_lock.lock(); - pointCloudBuf.push(laserCloudMsg); - mutex_lock.unlock(); -} - - -void laser_mapping(){ - while(1){ - if(!odometryBuf.empty() && !pointCloudBuf.empty()){ - - //read data - mutex_lock.lock(); - if(!pointCloudBuf.empty() && pointCloudBuf.front()->header.stamp.toSec()header.stamp.toSec()-0.5*lidar_param.scan_period){ - ROS_WARN("time stamp unaligned error and pointcloud discarded, pls check your data --> laser mapping node"); - pointCloudBuf.pop(); - mutex_lock.unlock(); - continue; - } - - if(!odometryBuf.empty() && odometryBuf.front()->header.stamp.toSec() < pointCloudBuf.front()->header.stamp.toSec()-0.5*lidar_param.scan_period){ - odometryBuf.pop(); - ROS_INFO("time stamp unaligned with path final, pls check your data --> laser mapping node"); - mutex_lock.unlock(); - continue; - } - - //if time aligned - pcl::PointCloud::Ptr pointcloud_in(new pcl::PointCloud()); - pcl::fromROSMsg(*pointCloudBuf.front(), *pointcloud_in); - ros::Time pointcloud_time = (pointCloudBuf.front())->header.stamp; - - Eigen::Isometry3d current_pose = Eigen::Isometry3d::Identity(); - current_pose.rotate(Eigen::Quaterniond(odometryBuf.front()->pose.pose.orientation.w,odometryBuf.front()->pose.pose.orientation.x,odometryBuf.front()->pose.pose.orientation.y,odometryBuf.front()->pose.pose.orientation.z)); - current_pose.pretranslate(Eigen::Vector3d(odometryBuf.front()->pose.pose.position.x,odometryBuf.front()->pose.pose.position.y,odometryBuf.front()->pose.pose.position.z)); - pointCloudBuf.pop(); - odometryBuf.pop(); - mutex_lock.unlock(); - - - laserMapping.updateCurrentPointsToMap(pointcloud_in,current_pose); - - pcl::PointCloud::Ptr pc_map = laserMapping.getMap(); - sensor_msgs::PointCloud2 PointsMsg; - pcl::toROSMsg(*pc_map, PointsMsg); - PointsMsg.header.stamp = pointcloud_time; - PointsMsg.header.frame_id = "map"; - map_pub.publish(PointsMsg); - - - - } - //sleep 2 ms every time - std::chrono::milliseconds dura(2); - std::this_thread::sleep_for(dura); - } -} - -int main(int argc, char **argv) -{ - ros::init(argc, argv, "main"); - ros::NodeHandle nh; - - int scan_line = 64; - double vertical_angle = 2.0; - double scan_period= 0.1; - double max_dis = 60.0; - double min_dis = 2.0; - double map_resolution = 0.4; - nh.getParam("/scan_period", scan_period); - nh.getParam("/vertical_angle", vertical_angle); - nh.getParam("/max_dis", max_dis); - nh.getParam("/min_dis", min_dis); - nh.getParam("/scan_line", scan_line); - nh.getParam("/map_resolution", map_resolution); - - lidar_param.setScanPeriod(scan_period); - lidar_param.setVerticalAngle(vertical_angle); - lidar_param.setLines(scan_line); - lidar_param.setMaxDistance(max_dis); - lidar_param.setMinDistance(min_dis); - - laserMapping.init(map_resolution); - ros::Subscriber subLaserCloud = nh.subscribe("/velodyne_points_filtered", 100, velodyneHandler); - ros::Subscriber subOdometry = nh.subscribe("/odom", 100, odomCallback); - - map_pub = nh.advertise("/map", 100); - std::thread laser_mapping_process{laser_mapping}; - - ros::spin(); - - return 0; -} diff --git a/src/laserProcessingClass.cpp b/src/laserProcessingClass.cpp deleted file mode 100644 index c8e1297..0000000 --- a/src/laserProcessingClass.cpp +++ /dev/null @@ -1,224 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro -#include "laserProcessingClass.h" - -void LaserProcessingClass::init(lidar::Lidar lidar_param_in){ - - lidar_param = lidar_param_in; - -} - -void LaserProcessingClass::featureExtraction(const pcl::PointCloud::Ptr& pc_in, pcl::PointCloud::Ptr& pc_out_edge, pcl::PointCloud::Ptr& pc_out_surf){ - - std::vector indices; - pcl::removeNaNFromPointCloud(*pc_in, indices); - - - int N_SCANS = lidar_param.num_lines; - std::vector::Ptr> laserCloudScans; - for(int i=0;i::Ptr(new pcl::PointCloud())); - } - - for (int i = 0; i < (int) pc_in->points.size(); i++) - { - int scanID=0; - double distance = sqrt(pc_in->points[i].x * pc_in->points[i].x + pc_in->points[i].y * pc_in->points[i].y); - if(distancelidar_param.max_distance) - continue; - double angle = atan(pc_in->points[i].z / distance) * 180 / M_PI; - - if (N_SCANS == 16) - { - scanID = int((angle + 15) / 2 + 0.5); - if (scanID > (N_SCANS - 1) || scanID < 0) - { - continue; - } - } - else if (N_SCANS == 32) - { - scanID = int((angle + 92.0/3.0) * 3.0 / 4.0); - if (scanID > (N_SCANS - 1) || scanID < 0) - { - continue; - } - } - else if (N_SCANS == 64) - { - if (angle >= -8.83) - scanID = int((2 - angle) * 3.0 + 0.5); - else - scanID = N_SCANS / 2 + int((-8.83 - angle) * 2.0 + 0.5); - - if (angle > 2 || angle < -24.33 || scanID > 63 || scanID < 0) - { - continue; - } - } - else - { - printf("wrong scan number\n"); - } - laserCloudScans[scanID]->push_back(pc_in->points[i]); - - } - - for(int i = 0; i < N_SCANS; i++){ - if(laserCloudScans[i]->points.size()<131){ - continue; - } - - std::vector cloudCurvature; - int total_points = laserCloudScans[i]->points.size()-10; - for(int j = 5; j < (int)laserCloudScans[i]->points.size() - 5; j++){ - double diffX = laserCloudScans[i]->points[j - 5].x + laserCloudScans[i]->points[j - 4].x + laserCloudScans[i]->points[j - 3].x + laserCloudScans[i]->points[j - 2].x + laserCloudScans[i]->points[j - 1].x - 10 * laserCloudScans[i]->points[j].x + laserCloudScans[i]->points[j + 1].x + laserCloudScans[i]->points[j + 2].x + laserCloudScans[i]->points[j + 3].x + laserCloudScans[i]->points[j + 4].x + laserCloudScans[i]->points[j + 5].x; - double diffY = laserCloudScans[i]->points[j - 5].y + laserCloudScans[i]->points[j - 4].y + laserCloudScans[i]->points[j - 3].y + laserCloudScans[i]->points[j - 2].y + laserCloudScans[i]->points[j - 1].y - 10 * laserCloudScans[i]->points[j].y + laserCloudScans[i]->points[j + 1].y + laserCloudScans[i]->points[j + 2].y + laserCloudScans[i]->points[j + 3].y + laserCloudScans[i]->points[j + 4].y + laserCloudScans[i]->points[j + 5].y; - double diffZ = laserCloudScans[i]->points[j - 5].z + laserCloudScans[i]->points[j - 4].z + laserCloudScans[i]->points[j - 3].z + laserCloudScans[i]->points[j - 2].z + laserCloudScans[i]->points[j - 1].z - 10 * laserCloudScans[i]->points[j].z + laserCloudScans[i]->points[j + 1].z + laserCloudScans[i]->points[j + 2].z + laserCloudScans[i]->points[j + 3].z + laserCloudScans[i]->points[j + 4].z + laserCloudScans[i]->points[j + 5].z; - Double2d distance(j,diffX * diffX + diffY * diffY + diffZ * diffZ); - cloudCurvature.push_back(distance); - - } - for(int j=0;j<6;j++){ - int sector_length = (int)(total_points/6); - int sector_start = sector_length *j; - int sector_end = sector_length *(j+1)-1; - if (j==5){ - sector_end = total_points - 1; - } - std::vector subCloudCurvature(cloudCurvature.begin()+sector_start,cloudCurvature.begin()+sector_end); - - featureExtractionFromSector(laserCloudScans[i],subCloudCurvature, pc_out_edge, pc_out_surf); - - } - - } - -} - - -void LaserProcessingClass::featureExtractionFromSector(const pcl::PointCloud::Ptr& pc_in, std::vector& cloudCurvature, pcl::PointCloud::Ptr& pc_out_edge, pcl::PointCloud::Ptr& pc_out_surf){ - - std::sort(cloudCurvature.begin(), cloudCurvature.end(), [](const Double2d & a, const Double2d & b) - { - return a.value < b.value; - }); - - - int largestPickedNum = 0; - std::vector picked_points; - int point_info_count =0; - for (int i = cloudCurvature.size()-1; i >= 0; i--) - { - int ind = cloudCurvature[i].id; - if(std::find(picked_points.begin(), picked_points.end(), ind)==picked_points.end()){ - if(cloudCurvature[i].value <= 0.1){ - break; - } - - largestPickedNum++; - picked_points.push_back(ind); - - if (largestPickedNum <= 20){ - pc_out_edge->push_back(pc_in->points[ind]); - point_info_count++; - }else{ - break; - } - - for(int k=1;k<=5;k++){ - double diffX = pc_in->points[ind + k].x - pc_in->points[ind + k - 1].x; - double diffY = pc_in->points[ind + k].y - pc_in->points[ind + k - 1].y; - double diffZ = pc_in->points[ind + k].z - pc_in->points[ind + k - 1].z; - if (diffX * diffX + diffY * diffY + diffZ * diffZ > 0.05){ - break; - } - picked_points.push_back(ind+k); - } - for(int k=-1;k>=-5;k--){ - double diffX = pc_in->points[ind + k].x - pc_in->points[ind + k + 1].x; - double diffY = pc_in->points[ind + k].y - pc_in->points[ind + k + 1].y; - double diffZ = pc_in->points[ind + k].z - pc_in->points[ind + k + 1].z; - if (diffX * diffX + diffY * diffY + diffZ * diffZ > 0.05){ - break; - } - picked_points.push_back(ind+k); - } - - } - } - - //find flat points - // point_info_count =0; - // int smallestPickedNum = 0; - - // for (int i = 0; i <= (int)cloudCurvature.size()-1; i++) - // { - // int ind = cloudCurvature[i].id; - - // if( std::find(picked_points.begin(), picked_points.end(), ind)==picked_points.end()){ - // if(cloudCurvature[i].value > 0.1){ - // //ROS_WARN("extracted feature not qualified, please check lidar"); - // break; - // } - // smallestPickedNum++; - // picked_points.push_back(ind); - - // if(smallestPickedNum <= 4){ - // //find all points - // pc_surf_flat->push_back(pc_in->points[ind]); - // pc_surf_lessFlat->push_back(pc_in->points[ind]); - // point_info_count++; - // } - // else{ - // break; - // } - - // for(int k=1;k<=5;k++){ - // double diffX = pc_in->points[ind + k].x - pc_in->points[ind + k - 1].x; - // double diffY = pc_in->points[ind + k].y - pc_in->points[ind + k - 1].y; - // double diffZ = pc_in->points[ind + k].z - pc_in->points[ind + k - 1].z; - // if (diffX * diffX + diffY * diffY + diffZ * diffZ > 0.05){ - // break; - // } - // picked_points.push_back(ind+k); - // } - // for(int k=-1;k>=-5;k--){ - // double diffX = pc_in->points[ind + k].x - pc_in->points[ind + k + 1].x; - // double diffY = pc_in->points[ind + k].y - pc_in->points[ind + k + 1].y; - // double diffZ = pc_in->points[ind + k].z - pc_in->points[ind + k + 1].z; - // if (diffX * diffX + diffY * diffY + diffZ * diffZ > 0.05){ - // break; - // } - // picked_points.push_back(ind+k); - // } - - // } - // } - - for (int i = 0; i <= (int)cloudCurvature.size()-1; i++) - { - int ind = cloudCurvature[i].id; - if( std::find(picked_points.begin(), picked_points.end(), ind)==picked_points.end()) - { - pc_out_surf->push_back(pc_in->points[ind]); - } - } - - - -} -LaserProcessingClass::LaserProcessingClass(){ - -} - -Double2d::Double2d(int id_in, double value_in){ - id = id_in; - value =value_in; -}; - -PointsInfo::PointsInfo(int layer_in, double time_in){ - layer = layer_in; - time = time_in; -}; diff --git a/src/laserProcessingNode.cpp b/src/laserProcessingNode.cpp deleted file mode 100644 index 0ec8564..0000000 --- a/src/laserProcessingNode.cpp +++ /dev/null @@ -1,143 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro - -//c++ lib -#include -#include -#include -#include -#include -#include - -//ros lib -#include -#include -#include -#include -#include -#include - -//pcl lib -#include -#include -#include - -//local lib -#include "lidar.h" -#include "laserProcessingClass.h" - - -LaserProcessingClass laserProcessing; -std::mutex mutex_lock; -std::queue pointCloudBuf; -lidar::Lidar lidar_param; - -ros::Publisher pubEdgePoints; -ros::Publisher pubSurfPoints; -ros::Publisher pubLaserCloudFiltered; - -void velodyneHandler(const sensor_msgs::PointCloud2ConstPtr &laserCloudMsg) -{ - mutex_lock.lock(); - pointCloudBuf.push(laserCloudMsg); - mutex_lock.unlock(); - -} - -double total_time =0; -int total_frame=0; - -void laser_processing(){ - while(1){ - if(!pointCloudBuf.empty()){ - //read data - mutex_lock.lock(); - pcl::PointCloud::Ptr pointcloud_in(new pcl::PointCloud()); - pcl::fromROSMsg(*pointCloudBuf.front(), *pointcloud_in); - ros::Time pointcloud_time = (pointCloudBuf.front())->header.stamp; - pointCloudBuf.pop(); - mutex_lock.unlock(); - - pcl::PointCloud::Ptr pointcloud_edge(new pcl::PointCloud()); - pcl::PointCloud::Ptr pointcloud_surf(new pcl::PointCloud()); - - std::chrono::time_point start, end; - start = std::chrono::system_clock::now(); - laserProcessing.featureExtraction(pointcloud_in,pointcloud_edge,pointcloud_surf); - end = std::chrono::system_clock::now(); - std::chrono::duration elapsed_seconds = end - start; - total_frame++; - float time_temp = elapsed_seconds.count() * 1000; - total_time+=time_temp; - //ROS_INFO("average laser processing time %f ms \n \n", total_time/total_frame); - - sensor_msgs::PointCloud2 laserCloudFilteredMsg; - pcl::PointCloud::Ptr pointcloud_filtered(new pcl::PointCloud()); - *pointcloud_filtered+=*pointcloud_edge; - *pointcloud_filtered+=*pointcloud_surf; - pcl::toROSMsg(*pointcloud_filtered, laserCloudFilteredMsg); - laserCloudFilteredMsg.header.stamp = pointcloud_time; - laserCloudFilteredMsg.header.frame_id = "base_link"; - pubLaserCloudFiltered.publish(laserCloudFilteredMsg); - - sensor_msgs::PointCloud2 edgePointsMsg; - pcl::toROSMsg(*pointcloud_edge, edgePointsMsg); - edgePointsMsg.header.stamp = pointcloud_time; - edgePointsMsg.header.frame_id = "base_link"; - pubEdgePoints.publish(edgePointsMsg); - - - sensor_msgs::PointCloud2 surfPointsMsg; - pcl::toROSMsg(*pointcloud_surf, surfPointsMsg); - surfPointsMsg.header.stamp = pointcloud_time; - surfPointsMsg.header.frame_id = "base_link"; - pubSurfPoints.publish(surfPointsMsg); - - } - //sleep 2 ms every time - std::chrono::milliseconds dura(2); - std::this_thread::sleep_for(dura); - } -} - -int main(int argc, char **argv) -{ - ros::init(argc, argv, "main"); - ros::NodeHandle nh; - - int scan_line = 64; - double vertical_angle = 2.0; - double scan_period= 0.1; - double max_dis = 60.0; - double min_dis = 2.0; - - nh.getParam("/scan_period", scan_period); - nh.getParam("/vertical_angle", vertical_angle); - nh.getParam("/max_dis", max_dis); - nh.getParam("/min_dis", min_dis); - nh.getParam("/scan_line", scan_line); - - lidar_param.setScanPeriod(scan_period); - lidar_param.setVerticalAngle(vertical_angle); - lidar_param.setLines(scan_line); - lidar_param.setMaxDistance(max_dis); - lidar_param.setMinDistance(min_dis); - - laserProcessing.init(lidar_param); - - ros::Subscriber subLaserCloud = nh.subscribe("/velodyne_points", 100, velodyneHandler); - - pubLaserCloudFiltered = nh.advertise("/velodyne_points_filtered", 100); - - pubEdgePoints = nh.advertise("/laser_cloud_edge", 100); - - pubSurfPoints = nh.advertise("/laser_cloud_surf", 100); - - std::thread laser_processing_process{laser_processing}; - - ros::spin(); - - return 0; -} - diff --git a/src/laser_mapping.cpp b/src/laser_mapping.cpp new file mode 100644 index 0000000..0fc5d27 --- /dev/null +++ b/src/laser_mapping.cpp @@ -0,0 +1,120 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// ros header +#include + +// c++ header +#include +#include + +// pcl header +#include +#include + +// local header +#include "floam/laser_mapping_node.hpp" + + +LaserMappingNode::LaserMappingNode() + : Node("laser_mapping_node") +{ + int scan_line = 64; + double scan_period = 0.1; + double vertical_angle = 2.0; + double max_dist = 60.0; + double min_dist = 2.0; + double map_resolution = 0.4; + + this->declare_parameter("scan_line", scan_line); + this->declare_parameter("scan_period", scan_period); + this->declare_parameter("vertical_angle", vertical_angle); + this->declare_parameter("max_dist", max_dist); + this->declare_parameter("min_dist", min_dist); + this->declare_parameter("map_resolution", map_resolution); + + // load from parameter if provided + scan_line = this->get_parameter("scan_line").get_parameter_value().get(); + scan_period = this->get_parameter("scan_period").get_parameter_value().get(); + vertical_angle = this->get_parameter("vertical_angle").get_parameter_value().get(); + max_dist = this->get_parameter("max_dist").get_parameter_value().get(); + min_dist = this->get_parameter("min_dist").get_parameter_value().get(); + map_resolution = this->get_parameter("map_resolution").get_parameter_value().get(); + + lidar_param_.setScanPeriod(scan_period); + lidar_param_.setVerticalAngle(vertical_angle); + lidar_param_.setLines(scan_line); + lidar_param_.setMaxDistance(max_dist); + lidar_param_.setMinDistance(min_dist); + + laserMapping_.init(map_resolution); + + subLaserCloud_ = this->create_subscription( + "velodyne_points_filtered", 100, std::bind(&LaserMappingNode::velodyneHandler, this, std::placeholders::_1)); + subOdometry_ = this->create_subscription( + "odom", 100, std::bind(&LaserMappingNode::odomCallback, this, std::placeholders::_1)); + + pubMap_ = this->create_publisher("map", 100); +} + +void LaserMappingNode::laser_mapping() +{ + while (1) { + if (!odometryBuf_.empty() && !pointCloudBuf_.empty()) { + // read data + mutex_lock_.lock(); + if (!pointCloudBuf_.empty() && pointCloudBuf_.front()->header.stamp.sec < odometryBuf_.front()->header.stamp.sec - 0.5*lidar_param_.scan_period) { + pointCloudBuf_.pop(); + RCLCPP_WARN(this->get_logger(), "time stamp unaligned error and pointcloud discarded, pls check your data --> laser mapping node"); + mutex_lock_.unlock(); + continue; + } + + if (!odometryBuf_.empty() && odometryBuf_.front()->header.stamp.sec < pointCloudBuf_.front()->header.stamp.sec - 0.5*lidar_param_.scan_period) { + odometryBuf_.pop(); + RCLCPP_WARN(this->get_logger(), "time stamp unaligned with path final, pls check your data --> laser mapping node"); + mutex_lock_.unlock(); + continue; + } + + // if time aligned + pcl::PointCloud::Ptr pointcloud_in(new pcl::PointCloud()); + pcl::fromROSMsg(*pointCloudBuf_.front(), *pointcloud_in); + rclcpp::Time pointcloud_time = (pointCloudBuf_.front())->header.stamp; + + Eigen::Isometry3d current_pose = Eigen::Isometry3d::Identity(); + current_pose.rotate(Eigen::Quaterniond(odometryBuf_.front()->pose.pose.orientation.w, odometryBuf_.front()->pose.pose.orientation.x, odometryBuf_.front()->pose.pose.orientation.y, odometryBuf_.front()->pose.pose.orientation.z)); + current_pose.pretranslate(Eigen::Vector3d(odometryBuf_.front()->pose.pose.position.x, odometryBuf_.front()->pose.pose.position.y, odometryBuf_.front()->pose.pose.position.z)); + pointCloudBuf_.pop(); + odometryBuf_.pop(); + mutex_lock_.unlock(); + + laserMapping_.updateCurrentPointsToMap(pointcloud_in, current_pose); + + pcl::PointCloud::Ptr pc_map = laserMapping_.getMap(); + sensor_msgs::msg::PointCloud2 PointsMsg; + pcl::toROSMsg(*pc_map, PointsMsg); + PointsMsg.header.stamp = pointcloud_time; + PointsMsg.header.frame_id = "map"; + pubMap_->publish(PointsMsg); + } + + //sleep 2 ms every time + std::chrono::milliseconds dura(2); + std::this_thread::sleep_for(dura); + } +} + +void LaserMappingNode::odomCallback(const nav_msgs::msg::Odometry::ConstSharedPtr odomMsg) +{ + mutex_lock_.lock(); + odometryBuf_.push(odomMsg); + mutex_lock_.unlock(); +} + +void LaserMappingNode::velodyneHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg) +{ + mutex_lock_.lock(); + pointCloudBuf_.push(laserCloudMsg); + mutex_lock_.unlock(); +} diff --git a/src/laser_mapping_class.cpp b/src/laser_mapping_class.cpp new file mode 100644 index 0000000..8daec77 --- /dev/null +++ b/src/laser_mapping_class.cpp @@ -0,0 +1,210 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// pcl header +#include + +// eigen header +#include + +// local header +#include "floam/laser_mapping_class.hpp" + + +void LaserMappingClass::init(double map_resolution) +{ + //init map. init can have real object, but future added block does not need + for (int i = 0; i < LASER_CELL_RANGE_HORIZONTAL * 2 + 1; i++) { + std::vector::Ptr>> map_height_temp; + for (int j = 0; j < LASER_CELL_RANGE_HORIZONTAL * 2 + 1; j++) { + std::vector::Ptr> map_depth_temp; + for (int k = 0; k < LASER_CELL_RANGE_VERTICAL * 2 + 1; k++) { + pcl::PointCloud::Ptr point_cloud_temp(new pcl::PointCloud); + map_depth_temp.push_back(point_cloud_temp); + } + map_height_temp.push_back(map_depth_temp); + } + map.push_back(map_height_temp); + } + + origin_in_map_x = LASER_CELL_RANGE_HORIZONTAL; + origin_in_map_y = LASER_CELL_RANGE_HORIZONTAL; + origin_in_map_z = LASER_CELL_RANGE_VERTICAL; + map_width = LASER_CELL_RANGE_HORIZONTAL*2+1; + map_height = LASER_CELL_RANGE_HORIZONTAL*2+1; + map_depth = LASER_CELL_RANGE_HORIZONTAL*2+1; + + // downsampling size + downSizeFilter.setLeafSize(map_resolution, map_resolution, map_resolution); +} + +// update points to map +void LaserMappingClass::updateCurrentPointsToMap(const pcl::PointCloud::Ptr& pc_in, const Eigen::Isometry3d& pose_current) +{ + int currentPosIdX = int(std::floor(pose_current.translation().x() / LASER_CELL_WIDTH + 0.5)) + origin_in_map_x; + int currentPosIdY = int(std::floor(pose_current.translation().y() / LASER_CELL_HEIGHT + 0.5)) + origin_in_map_y; + int currentPosIdZ = int(std::floor(pose_current.translation().z() / LASER_CELL_DEPTH + 0.5)) + origin_in_map_z; + + // check is submap is null + checkPoints(currentPosIdX,currentPosIdY,currentPosIdZ); + + pcl::PointCloud::Ptr transformed_pc(new pcl::PointCloud()); + pcl::transformPointCloud(*pc_in, *transformed_pc, pose_current.cast()); + + // save points + for (int i = 0; i < (int)transformed_pc->points.size(); i++) { + pcl::PointXYZI point_temp = transformed_pc->points[i]; + // for visualization only + point_temp.intensity = std::min(1.0 , std::max(pc_in->points[i].z+2.0, 0.0) / 5); + int currentPointIdX = int(std::floor(point_temp.x / LASER_CELL_WIDTH + 0.5)) + origin_in_map_x; + int currentPointIdY = int(std::floor(point_temp.y / LASER_CELL_HEIGHT + 0.5)) + origin_in_map_y; + int currentPointIdZ = int(std::floor(point_temp.z / LASER_CELL_DEPTH + 0.5)) + origin_in_map_z; + + map[currentPointIdX][currentPointIdY][currentPointIdZ]->push_back(point_temp); + } + + // filtering points + for (int i = currentPosIdX - LASER_CELL_RANGE_HORIZONTAL; i < currentPosIdX + LASER_CELL_RANGE_HORIZONTAL + 1; i++) { + for (int j = currentPosIdY - LASER_CELL_RANGE_HORIZONTAL; j < currentPosIdY + LASER_CELL_RANGE_HORIZONTAL + 1; j++) { + for (int k = currentPosIdZ - LASER_CELL_RANGE_VERTICAL; k < currentPosIdZ + LASER_CELL_RANGE_VERTICAL + 1; k++) { + downSizeFilter.setInputCloud(map[i][j][k]); + downSizeFilter.filter(*(map[i][j][k])); + } + } + } +} + +pcl::PointCloud::Ptr LaserMappingClass::getMap() +{ + pcl::PointCloud::Ptr laserCloudMap = pcl::PointCloud::Ptr(new pcl::PointCloud()); + for (int i = 0; i < map_width; i++) { + for (int j = 0; j < map_height; j++) { + for (int k = 0; k < map_depth; k++) { + if (map[i][j][k] != nullptr) { + *laserCloudMap += *(map[i][j][k]); + } + } + } + } + return laserCloudMap; +} + +void LaserMappingClass::addWidthCellNegative() +{ + std::vector::Ptr>> map_height_temp; + for (int j = 0; j < map_height; j++) { + std::vector::Ptr> map_depth_temp; + for(int k = 0; k < map_depth; k++) { + pcl::PointCloud::Ptr point_cloud_temp; + map_depth_temp.push_back(point_cloud_temp); + } + map_height_temp.push_back(map_depth_temp); + } + map.insert(map.begin(), map_height_temp); + + origin_in_map_x++; + map_width++; +} + +void LaserMappingClass::addWidthCellPositive() +{ + std::vector::Ptr>> map_height_temp; + for (int j = 0; j < map_height; j++) { + std::vector::Ptr> map_depth_temp; + for (int k = 0; k < map_depth; k++) { + pcl::PointCloud::Ptr point_cloud_temp; + map_depth_temp.push_back(point_cloud_temp); + } + map_height_temp.push_back(map_depth_temp); + } + map.push_back(map_height_temp); + map_width++; +} + +void LaserMappingClass::addHeightCellNegative() +{ + for (int i = 0; i < map_width; i++) { + std::vector::Ptr> map_depth_temp; + for (int k=0; k < map_depth; k++) { + pcl::PointCloud::Ptr point_cloud_temp; + map_depth_temp.push_back(point_cloud_temp); + } + map[i].insert(map[i].begin(), map_depth_temp); + } + origin_in_map_y++; + map_height++; +} + +void LaserMappingClass::addHeightCellPositive() +{ + for (int i = 0; i < map_width; i++) { + std::vector::Ptr> map_depth_temp; + for (int k = 0; k < map_depth; k++) { + pcl::PointCloud::Ptr point_cloud_temp; + map_depth_temp.push_back(point_cloud_temp); + } + map[i].push_back(map_depth_temp); + } + map_height++; +} + +void LaserMappingClass::addDepthCellNegative() +{ + for (int i = 0; i < map_width; i++) { + for (int j = 0; j < map_height; j++) { + pcl::PointCloud::Ptr point_cloud_temp; + map[i][j].insert(map[i][j].begin(), point_cloud_temp); + } + } + origin_in_map_z++; + map_depth++; +} + +void LaserMappingClass::addDepthCellPositive() +{ + for (int i = 0; i < map_width; i++) { + for (int j = 0; j < map_height; j++) { + pcl::PointCloud::Ptr point_cloud_temp; + map[i][j].push_back(point_cloud_temp); + } + } + map_depth++; +} + +// extend map is points exceed size +void LaserMappingClass::checkPoints(int& x, int& y, int& z) +{ + while (x + LASER_CELL_RANGE_HORIZONTAL > map_width - 1) { + addWidthCellPositive(); + } + while (x - LASER_CELL_RANGE_HORIZONTAL < 0) { + addWidthCellNegative(); + x++; + } + while (y + LASER_CELL_RANGE_HORIZONTAL > map_height - 1) { + addHeightCellPositive(); + } + while (y - LASER_CELL_RANGE_HORIZONTAL < 0) { + addHeightCellNegative(); + y++; + } + while (z + LASER_CELL_RANGE_VERTICAL> map_depth - 1) { + addDepthCellPositive(); + } + while (z - LASER_CELL_RANGE_VERTICAL < 0) { + addDepthCellNegative(); + z++; + } + + // initialize. create object if area is null + for (int i = x - LASER_CELL_RANGE_HORIZONTAL; i < x + LASER_CELL_RANGE_HORIZONTAL + 1; i++) { + for (int j = y - LASER_CELL_RANGE_HORIZONTAL; j < y + LASER_CELL_RANGE_HORIZONTAL + 1; j++) { + for (int k = z - LASER_CELL_RANGE_VERTICAL; k < z + LASER_CELL_RANGE_VERTICAL+1; k++) { + if (map[i][j][k] == nullptr) { + pcl::PointCloud::Ptr point_cloud_temp(new pcl::PointCloud()); + map[i][j][k] = point_cloud_temp; + } + } + } + } +} diff --git a/src/laser_mapping_node.cpp b/src/laser_mapping_node.cpp new file mode 100644 index 0000000..686bd23 --- /dev/null +++ b/src/laser_mapping_node.cpp @@ -0,0 +1,23 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// ros header +#include + +// c++ header +#include + +// local header +#include "floam/laser_mapping_node.hpp" + + +int main(int argc, char* argv[]) +{ + rclcpp::init(argc, argv); + auto lmo_ptr {std::make_shared()}; + std::thread laser_mapping_process(&LaserMappingNode::laser_mapping, lmo_ptr); + rclcpp::spin(lmo_ptr); + rclcpp::shutdown(); + + return 0; +} diff --git a/src/laser_processing.cpp b/src/laser_processing.cpp new file mode 100644 index 0000000..6612ea1 --- /dev/null +++ b/src/laser_processing.cpp @@ -0,0 +1,111 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// ros header +#include + +// c++ header +#include +#include + +// local header +#include "floam/laser_processing_node.hpp" + + +LaserProcessingNode::LaserProcessingNode() + : Node("laser_processing_node"), total_time_(0.0), total_frame_(0) +{ + // set the default value of the parameters + int scan_line = 64; + double scan_period = 0.1; + double vertical_angle = 2.0; + double max_dist = 60.0; + double min_dist = 2.0; + this->declare_parameter("scan_line", scan_line); + this->declare_parameter("scan_period", scan_period); + this->declare_parameter("vertical_angle", vertical_angle); + this->declare_parameter("max_dist", max_dist); + this->declare_parameter("min_dist", min_dist); + + // load from parameter if provided + scan_line = this->get_parameter("scan_line").get_parameter_value().get(); + scan_period = this->get_parameter("scan_period").get_parameter_value().get(); + vertical_angle = this->get_parameter("vertical_angle").get_parameter_value().get(); + max_dist = this->get_parameter("max_dist").get_parameter_value().get(); + min_dist = this->get_parameter("min_dist").get_parameter_value().get(); + + lidar_param_.setScanPeriod(scan_period); + lidar_param_.setVerticalAngle(vertical_angle); + lidar_param_.setLines(scan_line); + lidar_param_.setMaxDistance(max_dist); + lidar_param_.setMinDistance(min_dist); + + laserProcessing_.init(lidar_param_); + + subLaserCloud_ = this->create_subscription( + "velodyne_points", 100, std::bind(&LaserProcessingNode::velodyneHandler, this, std::placeholders::_1)); + + pubLaserCloudFiltered_ = this->create_publisher("velodyne_points_filtered", 100); + pubEdgePoints_ = this->create_publisher("laser_cloud_edge", 100); + pubSurfPoints_ = this->create_publisher("laser_cloud_surf", 100); +} + +void LaserProcessingNode::laser_processing() +{ + while (1) { + if (!pointCloudBuf_.empty()) { + //read data + mutex_lock_.lock(); + pcl::PointCloud::Ptr pointcloud_in(new pcl::PointCloud()); + pcl::fromROSMsg(*pointCloudBuf_.front(), *pointcloud_in); + rclcpp::Time pointcloud_time = (pointCloudBuf_.front())->header.stamp; + pointCloudBuf_.pop(); + mutex_lock_.unlock(); + + pcl::PointCloud::Ptr pointcloud_edge(new pcl::PointCloud()); + pcl::PointCloud::Ptr pointcloud_surf(new pcl::PointCloud()); + + std::chrono::time_point start, end; + start = std::chrono::system_clock::now(); + laserProcessing_.featureExtraction(pointcloud_in, pointcloud_edge, pointcloud_surf); + end = std::chrono::system_clock::now(); + std::chrono::duration elapsed_seconds = end - start; + total_frame_++; + float time_temp = elapsed_seconds.count() * 1000; + total_time_ += time_temp; + //RCLCPP_INFO(this->get_logger(), "Average laser processing time %f ms \n \n", total_time_/total_frame_); + + sensor_msgs::msg::PointCloud2 laserCloudFilteredMsg; + pcl::PointCloud::Ptr pointcloud_filtered(new pcl::PointCloud()); + *pointcloud_filtered += *pointcloud_edge; + *pointcloud_filtered += *pointcloud_surf; + pcl::toROSMsg(*pointcloud_filtered, laserCloudFilteredMsg); + laserCloudFilteredMsg.header.stamp = pointcloud_time; + laserCloudFilteredMsg.header.frame_id = "base_link"; + pubLaserCloudFiltered_->publish(laserCloudFilteredMsg); + + sensor_msgs::msg::PointCloud2 edgePointsMsg; + pcl::toROSMsg(*pointcloud_edge, edgePointsMsg); + edgePointsMsg.header.stamp = pointcloud_time; + edgePointsMsg.header.frame_id = "base_link"; + pubEdgePoints_->publish(edgePointsMsg); + + sensor_msgs::msg::PointCloud2 surfPointsMsg; + pcl::toROSMsg(*pointcloud_surf, surfPointsMsg); + surfPointsMsg.header.stamp = pointcloud_time; + surfPointsMsg.header.frame_id = "base_link"; + pubSurfPoints_->publish(surfPointsMsg); + } + + //sleep 2 ms every time + std::chrono::milliseconds dura(2); + std::this_thread::sleep_for(dura); + } +} + +void LaserProcessingNode::velodyneHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg) +{ + mutex_lock_.lock(); + pointCloudBuf_.push(laserCloudMsg); + mutex_lock_.unlock(); +} diff --git a/src/laser_processing_class.cpp b/src/laser_processing_class.cpp new file mode 100644 index 0000000..dd0ed2e --- /dev/null +++ b/src/laser_processing_class.cpp @@ -0,0 +1,167 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// c++ header +#include +#include + +// pcl header +#include + +// local header +#include "floam/laser_processing_class.hpp" + + +Double2d::Double2d(int id_in, double value_in) +{ + id = id_in; + value =value_in; +} + +PointsInfo::PointsInfo(int layer_in, double time_in) +{ + layer = layer_in; + time = time_in; +} + +void LaserProcessingClass::init(lidar::Lidar lidar_param_in) +{ + lidar_param = lidar_param_in; +} + +void LaserProcessingClass::featureExtraction(const pcl::PointCloud::Ptr& pc_in, + pcl::PointCloud::Ptr& pc_out_edge, + pcl::PointCloud::Ptr& pc_out_surf) +{ + std::vector indices; + pcl::removeNaNFromPointCloud(*pc_in, *pc_in, indices); + + int N_SCANS = lidar_param.num_lines; + std::vector::Ptr> laserCloudScans; + + for (int i = 0; i < N_SCANS; ++i) { + laserCloudScans.push_back(pcl::PointCloud::Ptr(new pcl::PointCloud())); + } + + for (int i = 0; i < (int)pc_in->points.size(); i++) { + int scanID = 0; + double distance = sqrt(pc_in->points[i].x * pc_in->points[i].x + pc_in->points[i].y * pc_in->points[i].y); + if (distance < lidar_param.min_distance || distance > lidar_param.max_distance) { + continue; + } + + double angle = atan(pc_in->points[i].z / distance) * 180 / M_PI; + + if (N_SCANS == 16) { + scanID = int((angle + 15) / 2 + 0.5); + if (scanID > (N_SCANS - 1) || scanID < 0) { + continue; + } + } else if (N_SCANS == 32) { + scanID = int((angle + 92.0/3.0) * 3.0 / 4.0); + if (scanID > (N_SCANS - 1) || scanID < 0) { + continue; + } + } else if (N_SCANS == 64) { + if (angle >= -8.83) { + scanID = int((2 - angle) * 3.0 + 0.5); + } else { + scanID = N_SCANS / 2 + int((-8.83 - angle) * 2.0 + 0.5); + } + + if (angle > 2 || angle < -24.33 || scanID > 63 || scanID < 0) { + continue; + } + } else { + printf("wrong scan number\n"); + } + + laserCloudScans[scanID]->push_back(pc_in->points[i]); + } + + for (int i = 0; i < N_SCANS; i++) { + if (laserCloudScans[i]->points.size() < 131) { + continue; + } + + std::vector cloudCurvature; + int total_points = laserCloudScans[i]->points.size() - 10; + for (int j = 5; j < (int)laserCloudScans[i]->points.size() - 5; ++j) { + double diffX = laserCloudScans[i]->points[j - 5].x + laserCloudScans[i]->points[j - 4].x + laserCloudScans[i]->points[j - 3].x + laserCloudScans[i]->points[j - 2].x + laserCloudScans[i]->points[j - 1].x - 10 * laserCloudScans[i]->points[j].x + laserCloudScans[i]->points[j + 1].x + laserCloudScans[i]->points[j + 2].x + laserCloudScans[i]->points[j + 3].x + laserCloudScans[i]->points[j + 4].x + laserCloudScans[i]->points[j + 5].x; + double diffY = laserCloudScans[i]->points[j - 5].y + laserCloudScans[i]->points[j - 4].y + laserCloudScans[i]->points[j - 3].y + laserCloudScans[i]->points[j - 2].y + laserCloudScans[i]->points[j - 1].y - 10 * laserCloudScans[i]->points[j].y + laserCloudScans[i]->points[j + 1].y + laserCloudScans[i]->points[j + 2].y + laserCloudScans[i]->points[j + 3].y + laserCloudScans[i]->points[j + 4].y + laserCloudScans[i]->points[j + 5].y; + double diffZ = laserCloudScans[i]->points[j - 5].z + laserCloudScans[i]->points[j - 4].z + laserCloudScans[i]->points[j - 3].z + laserCloudScans[i]->points[j - 2].z + laserCloudScans[i]->points[j - 1].z - 10 * laserCloudScans[i]->points[j].z + laserCloudScans[i]->points[j + 1].z + laserCloudScans[i]->points[j + 2].z + laserCloudScans[i]->points[j + 3].z + laserCloudScans[i]->points[j + 4].z + laserCloudScans[i]->points[j + 5].z; + Double2d distance(j,diffX * diffX + diffY * diffY + diffZ * diffZ); + cloudCurvature.push_back(distance); + } + + for (int j = 0; j < 6; ++j) { + int sector_length = (int)(total_points / 6); + int sector_start = sector_length * j; + int sector_end = sector_length * (j+1) - 1; + if (j==5) { + sector_end = total_points - 1; + } + std::vector subCloudCurvature(cloudCurvature.begin()+sector_start,cloudCurvature.begin()+sector_end); + + featureExtractionFromSector(laserCloudScans[i],subCloudCurvature, pc_out_edge, pc_out_surf); + } + } +} + +void LaserProcessingClass::featureExtractionFromSector(const pcl::PointCloud::Ptr& pc_in, + std::vector& cloudCurvature, + pcl::PointCloud::Ptr& pc_out_edge, + pcl::PointCloud::Ptr& pc_out_surf) +{ + std::sort(cloudCurvature.begin(), cloudCurvature.end(), [](const Double2d & a, const Double2d & b) { + return a.value < b.value; + }); + + int largestPickedNum = 0; + std::vector picked_points; + int point_info_count = 0; + for (int i = cloudCurvature.size()-1; i >= 0; i--) { + int ind = cloudCurvature[i].id; + if (std::find(picked_points.begin(), picked_points.end(), ind)==picked_points.end()) { + if (cloudCurvature[i].value <= 0.1) { + break; + } + + largestPickedNum++; + picked_points.push_back(ind); + + if (largestPickedNum <= 20) { + pc_out_edge->push_back(pc_in->points[ind]); + point_info_count++; + } else { + break; + } + + for (int k=1; k<=5; k++) { + double diffX = pc_in->points[ind + k].x - pc_in->points[ind + k - 1].x; + double diffY = pc_in->points[ind + k].y - pc_in->points[ind + k - 1].y; + double diffZ = pc_in->points[ind + k].z - pc_in->points[ind + k - 1].z; + if (diffX * diffX + diffY * diffY + diffZ * diffZ > 0.05) { + break; + } + picked_points.push_back(ind+k); + } + for (int k=-1; k>=-5; k--) { + double diffX = pc_in->points[ind + k].x - pc_in->points[ind + k + 1].x; + double diffY = pc_in->points[ind + k].y - pc_in->points[ind + k + 1].y; + double diffZ = pc_in->points[ind + k].z - pc_in->points[ind + k + 1].z; + if (diffX * diffX + diffY * diffY + diffZ * diffZ > 0.05) { + break; + } + picked_points.push_back(ind+k); + } + } + } + + for (int i = 0; i < (int)cloudCurvature.size(); i++) { + int ind = cloudCurvature[i].id; + if (std::find(picked_points.begin(), picked_points.end(), ind) == picked_points.end()) { + pc_out_surf->push_back(pc_in->points[ind]); + } + } +} diff --git a/src/laser_processing_node.cpp b/src/laser_processing_node.cpp new file mode 100644 index 0000000..180fda9 --- /dev/null +++ b/src/laser_processing_node.cpp @@ -0,0 +1,23 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// ros header +#include + +// c++ header +#include + +// local header +#include "floam/laser_processing_node.hpp" + + +int main(int argc, char* argv[]) +{ + rclcpp::init(argc, argv); + auto lpn_ptr {std::make_shared()}; + std::thread laser_processing_process(&LaserProcessingNode::laser_processing, lpn_ptr); + rclcpp::spin(lpn_ptr); + rclcpp::shutdown(); + + return 0; +} diff --git a/src/lidar.cpp b/src/lidar.cpp index 4e73451..f9d9a46 100644 --- a/src/lidar.cpp +++ b/src/lidar.cpp @@ -1,39 +1,36 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang -#include "lidar.h" +// local header +#include "floam/lidar.hpp" -lidar::Lidar::Lidar(){ - +void lidar::Lidar::setLines(int num_lines_in) +{ + num_lines=num_lines_in; } - -void lidar::Lidar::setLines(double num_lines_in){ - num_lines=num_lines_in; +void lidar::Lidar::setVerticalAngle(double vertical_angle_in) +{ + vertical_angle = vertical_angle_in; } - -void lidar::Lidar::setVerticalAngle(double vertical_angle_in){ - vertical_angle = vertical_angle_in; +void lidar::Lidar::setVerticalResolution(double vertical_angle_resolution_in) +{ + vertical_angle_resolution = vertical_angle_resolution_in; } - -void lidar::Lidar::setVerticalResolution(double vertical_angle_resolution_in){ - vertical_angle_resolution = vertical_angle_resolution_in; +void lidar::Lidar::setScanPeriod(double scan_period_in) +{ + scan_period = scan_period_in; } - -void lidar::Lidar::setScanPeriod(double scan_period_in){ - scan_period = scan_period_in; +void lidar::Lidar::setMaxDistance(double max_distance_in) +{ + max_distance = max_distance_in; } - -void lidar::Lidar::setMaxDistance(double max_distance_in){ - max_distance = max_distance_in; +void lidar::Lidar::setMinDistance(double min_distance_in) +{ + min_distance = min_distance_in; } - -void lidar::Lidar::setMinDistance(double min_distance_in){ - min_distance = min_distance_in; -} \ No newline at end of file diff --git a/src/lidarOptimization.cpp b/src/lidarOptimization.cpp deleted file mode 100644 index 64cc0f1..0000000 --- a/src/lidarOptimization.cpp +++ /dev/null @@ -1,152 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro - -#include "lidarOptimization.h" - -EdgeAnalyticCostFunction::EdgeAnalyticCostFunction(Eigen::Vector3d curr_point_, Eigen::Vector3d last_point_a_, Eigen::Vector3d last_point_b_) - : curr_point(curr_point_), last_point_a(last_point_a_), last_point_b(last_point_b_){ - -} - -bool EdgeAnalyticCostFunction::Evaluate(double const *const *parameters, double *residuals, double **jacobians) const -{ - - Eigen::Map q_last_curr(parameters[0]); - Eigen::Map t_last_curr(parameters[0] + 4); - Eigen::Vector3d lp; - lp = q_last_curr * curr_point + t_last_curr; - - Eigen::Vector3d nu = (lp - last_point_a).cross(lp - last_point_b); - Eigen::Vector3d de = last_point_a - last_point_b; - double de_norm = de.norm(); - residuals[0] = nu.norm()/de_norm; - - if(jacobians != NULL) - { - if(jacobians[0] != NULL) - { - Eigen::Matrix3d skew_lp = skew(lp); - Eigen::Matrix dp_by_se3; - dp_by_se3.block<3,3>(0,0) = -skew_lp; - (dp_by_se3.block<3,3>(0, 3)).setIdentity(); - Eigen::Map > J_se3(jacobians[0]); - J_se3.setZero(); - Eigen::Matrix3d skew_de = skew(de); - J_se3.block<1,6>(0,0) = - nu.transpose() / nu.norm() * skew_de * dp_by_se3/de_norm; - - } - } - - return true; - -} - - -SurfNormAnalyticCostFunction::SurfNormAnalyticCostFunction(Eigen::Vector3d curr_point_, Eigen::Vector3d plane_unit_norm_, double negative_OA_dot_norm_) - : curr_point(curr_point_), plane_unit_norm(plane_unit_norm_), negative_OA_dot_norm(negative_OA_dot_norm_){ - -} - -bool SurfNormAnalyticCostFunction::Evaluate(double const *const *parameters, double *residuals, double **jacobians) const -{ - Eigen::Map q_w_curr(parameters[0]); - Eigen::Map t_w_curr(parameters[0] + 4); - Eigen::Vector3d point_w = q_w_curr * curr_point + t_w_curr; - residuals[0] = plane_unit_norm.dot(point_w) + negative_OA_dot_norm; - - if(jacobians != NULL) - { - if(jacobians[0] != NULL) - { - Eigen::Matrix3d skew_point_w = skew(point_w); - Eigen::Matrix dp_by_se3; - dp_by_se3.block<3,3>(0,0) = -skew_point_w; - (dp_by_se3.block<3,3>(0, 3)).setIdentity(); - Eigen::Map > J_se3(jacobians[0]); - J_se3.setZero(); - J_se3.block<1,6>(0,0) = plane_unit_norm.transpose() * dp_by_se3; - - } - } - return true; - -} - - -bool PoseSE3Parameterization::Plus(const double *x, const double *delta, double *x_plus_delta) const -{ - Eigen::Map trans(x + 4); - - Eigen::Quaterniond delta_q; - Eigen::Vector3d delta_t; - getTransformFromSe3(Eigen::Map>(delta), delta_q, delta_t); - Eigen::Map quater(x); - Eigen::Map quater_plus(x_plus_delta); - Eigen::Map trans_plus(x_plus_delta + 4); - - quater_plus = delta_q * quater; - trans_plus = delta_q * trans + delta_t; - - return true; -} - -bool PoseSE3Parameterization::ComputeJacobian(const double *x, double *jacobian) const -{ - Eigen::Map> j(jacobian); - (j.topRows(6)).setIdentity(); - (j.bottomRows(1)).setZero(); - - return true; -} - -void getTransformFromSe3(const Eigen::Matrix& se3, Eigen::Quaterniond& q, Eigen::Vector3d& t){ - Eigen::Vector3d omega(se3.data()); - Eigen::Vector3d upsilon(se3.data()+3); - Eigen::Matrix3d Omega = skew(omega); - - double theta = omega.norm(); - double half_theta = 0.5*theta; - - double imag_factor; - double real_factor = cos(half_theta); - if(theta<1e-10) - { - double theta_sq = theta*theta; - double theta_po4 = theta_sq*theta_sq; - imag_factor = 0.5-0.0208333*theta_sq+0.000260417*theta_po4; - } - else - { - double sin_half_theta = sin(half_theta); - imag_factor = sin_half_theta/theta; - } - - q = Eigen::Quaterniond(real_factor, imag_factor*omega.x(), imag_factor*omega.y(), imag_factor*omega.z()); - - - Eigen::Matrix3d J; - if (theta<1e-10) - { - J = q.matrix(); - } - else - { - Eigen::Matrix3d Omega2 = Omega*Omega; - J = (Eigen::Matrix3d::Identity() + (1-cos(theta))/(theta*theta)*Omega + (theta-sin(theta))/(pow(theta,3))*Omega2); - } - - t = J*upsilon; -} - -Eigen::Matrix skew(Eigen::Matrix& mat_in){ - Eigen::Matrix skew_mat; - skew_mat.setZero(); - skew_mat(0,1) = -mat_in(2); - skew_mat(0,2) = mat_in(1); - skew_mat(1,2) = -mat_in(0); - skew_mat(1,0) = mat_in(2); - skew_mat(2,0) = -mat_in(1); - skew_mat(2,1) = mat_in(0); - return skew_mat; -} \ No newline at end of file diff --git a/src/lidar_optimization.cpp b/src/lidar_optimization.cpp new file mode 100644 index 0000000..5ec9c63 --- /dev/null +++ b/src/lidar_optimization.cpp @@ -0,0 +1,141 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// local header +#include "floam/lidar_optimization.hpp" + + +void getTransformFromSe3(const Eigen::Matrix& se3, Eigen::Quaterniond& q, Eigen::Vector3d& t) +{ + Eigen::Vector3d omega(se3.data()); + Eigen::Vector3d upsilon(se3.data()+3); + Eigen::Matrix3d Omega = skew(omega); + + double theta = omega.norm(); + double half_theta = 0.5*theta; + + double imag_factor; + double real_factor = cos(half_theta); + if (theta<1e-10) { + double theta_sq = theta*theta; + double theta_po4 = theta_sq*theta_sq; + imag_factor = 0.5-0.0208333*theta_sq+0.000260417*theta_po4; + } else { + double sin_half_theta = sin(half_theta); + imag_factor = sin_half_theta/theta; + } + + q = Eigen::Quaterniond(real_factor, imag_factor*omega.x(), imag_factor*omega.y(), imag_factor*omega.z()); + + Eigen::Matrix3d J; + if (theta<1e-10) { + J = q.matrix(); + } else { + Eigen::Matrix3d Omega2 = Omega*Omega; + J = (Eigen::Matrix3d::Identity() + (1-cos(theta))/(theta*theta)*Omega + (theta-sin(theta))/(pow(theta,3))*Omega2); + } + t = J*upsilon; +} + +Eigen::Matrix skew(Eigen::Matrix& mat_in) +{ + Eigen::Matrix skew_mat; + skew_mat.setZero(); + skew_mat(0,1) = -mat_in(2); + skew_mat(0,2) = mat_in(1); + skew_mat(1,2) = -mat_in(0); + skew_mat(1,0) = mat_in(2); + skew_mat(2,0) = -mat_in(1); + skew_mat(2,1) = mat_in(0); + + return skew_mat; +} + +EdgeAnalyticCostFunction::EdgeAnalyticCostFunction(Eigen::Vector3d curr_point, + Eigen::Vector3d last_point_a, Eigen::Vector3d last_point_b) + : curr_point_(curr_point), last_point_a_(last_point_a), last_point_b_(last_point_b) +{ +} + +bool EdgeAnalyticCostFunction::Evaluate(double const *const *parameters, + double *residuals, double **jacobians) const +{ + Eigen::Map q_last_curr(parameters[0]); + Eigen::Map t_last_curr(parameters[0] + 4); + Eigen::Vector3d lp; + lp = q_last_curr * curr_point_ + t_last_curr; + + Eigen::Vector3d nu = (lp - last_point_a_).cross(lp - last_point_b_); + Eigen::Vector3d de = last_point_a_ - last_point_b_; + double de_norm = de.norm(); + residuals[0] = nu.norm()/de_norm; + + if (jacobians != nullptr) { + if (jacobians[0] != nullptr) { + Eigen::Matrix3d skew_lp = skew(lp); + Eigen::Matrix dp_by_se3; + dp_by_se3.block<3,3>(0,0) = -skew_lp; + (dp_by_se3.block<3,3>(0, 3)).setIdentity(); + Eigen::Map > J_se3(jacobians[0]); + J_se3.setZero(); + Eigen::Matrix3d skew_de = skew(de); + J_se3.block<1,6>(0,0) = - nu.transpose() / nu.norm() * skew_de * dp_by_se3/de_norm; + } + } + + return true; +} + +SurfNormAnalyticCostFunction::SurfNormAnalyticCostFunction(Eigen::Vector3d curr_point, + Eigen::Vector3d plane_unit_norm, double negative_OA_dot_norm) + : curr_point_(curr_point), plane_unit_norm_(plane_unit_norm), negative_OA_dot_norm_(negative_OA_dot_norm) +{ +} + +bool SurfNormAnalyticCostFunction::Evaluate(double const *const *parameters, double *residuals, double **jacobians) const +{ + Eigen::Map q_w_curr(parameters[0]); + Eigen::Map t_w_curr(parameters[0] + 4); + Eigen::Vector3d point_w = q_w_curr * curr_point_ + t_w_curr; + residuals[0] = plane_unit_norm_.dot(point_w) + negative_OA_dot_norm_; + + if (jacobians != nullptr) { + if (jacobians[0] != nullptr) { + Eigen::Matrix3d skew_point_w = skew(point_w); + Eigen::Matrix dp_by_se3; + dp_by_se3.block<3,3>(0,0) = -skew_point_w; + (dp_by_se3.block<3,3>(0, 3)).setIdentity(); + Eigen::Map > J_se3(jacobians[0]); + J_se3.setZero(); + J_se3.block<1,6>(0,0) = plane_unit_norm_.transpose() * dp_by_se3; + } + } + + return true; +} + +bool PoseSE3Parameterization::Plus(const double *x, const double *delta, double *x_plus_delta) const +{ + Eigen::Map trans(x + 4); + + Eigen::Quaterniond delta_q; + Eigen::Vector3d delta_t; + getTransformFromSe3(Eigen::Map>(delta), delta_q, delta_t); + Eigen::Map quater(x); + Eigen::Map quater_plus(x_plus_delta); + Eigen::Map trans_plus(x_plus_delta + 4); + + quater_plus = delta_q * quater; + trans_plus = delta_q * trans + delta_t; + + return true; +} + +bool PoseSE3Parameterization::ComputeJacobian(const double *x, double *jacobian) const +{ + Eigen::Map> j(jacobian); + (j.topRows(6)).setIdentity(); + (j.bottomRows(1)).setZero(); + + return true; +} diff --git a/src/odomEstimationClass.cpp b/src/odomEstimationClass.cpp deleted file mode 100644 index f14e639..0000000 --- a/src/odomEstimationClass.cpp +++ /dev/null @@ -1,260 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro - -#include "odomEstimationClass.h" - -void OdomEstimationClass::init(lidar::Lidar lidar_param, double map_resolution){ - //init local map - laserCloudCornerMap = pcl::PointCloud::Ptr(new pcl::PointCloud()); - laserCloudSurfMap = pcl::PointCloud::Ptr(new pcl::PointCloud()); - - //downsampling size - downSizeFilterEdge.setLeafSize(map_resolution, map_resolution, map_resolution); - downSizeFilterSurf.setLeafSize(map_resolution * 2, map_resolution * 2, map_resolution * 2); - - //kd-tree - kdtreeEdgeMap = pcl::KdTreeFLANN::Ptr(new pcl::KdTreeFLANN()); - kdtreeSurfMap = pcl::KdTreeFLANN::Ptr(new pcl::KdTreeFLANN()); - - odom = Eigen::Isometry3d::Identity(); - last_odom = Eigen::Isometry3d::Identity(); - optimization_count=2; -} - -void OdomEstimationClass::initMapWithPoints(const pcl::PointCloud::Ptr& edge_in, const pcl::PointCloud::Ptr& surf_in){ - *laserCloudCornerMap += *edge_in; - *laserCloudSurfMap += *surf_in; - optimization_count=12; -} - - -void OdomEstimationClass::updatePointsToMap(const pcl::PointCloud::Ptr& edge_in, const pcl::PointCloud::Ptr& surf_in){ - - if(optimization_count>2) - optimization_count--; - - Eigen::Isometry3d odom_prediction = odom * (last_odom.inverse() * odom); - last_odom = odom; - odom = odom_prediction; - - q_w_curr = Eigen::Quaterniond(odom.rotation()); - t_w_curr = odom.translation(); - - pcl::PointCloud::Ptr downsampledEdgeCloud(new pcl::PointCloud()); - pcl::PointCloud::Ptr downsampledSurfCloud(new pcl::PointCloud()); - downSamplingToMap(edge_in,downsampledEdgeCloud,surf_in,downsampledSurfCloud); - //ROS_WARN("point nyum%d,%d",(int)downsampledEdgeCloud->points.size(), (int)downsampledSurfCloud->points.size()); - if(laserCloudCornerMap->points.size()>10 && laserCloudSurfMap->points.size()>50){ - kdtreeEdgeMap->setInputCloud(laserCloudCornerMap); - kdtreeSurfMap->setInputCloud(laserCloudSurfMap); - - for (int iterCount = 0; iterCount < optimization_count; iterCount++){ - ceres::LossFunction *loss_function = new ceres::HuberLoss(0.1); - ceres::Problem::Options problem_options; - ceres::Problem problem(problem_options); - - problem.AddParameterBlock(parameters, 7, new PoseSE3Parameterization()); - - addEdgeCostFactor(downsampledEdgeCloud,laserCloudCornerMap,problem,loss_function); - addSurfCostFactor(downsampledSurfCloud,laserCloudSurfMap,problem,loss_function); - - ceres::Solver::Options options; - options.linear_solver_type = ceres::DENSE_QR; - options.max_num_iterations = 4; - options.minimizer_progress_to_stdout = false; - options.check_gradients = false; - options.gradient_check_relative_precision = 1e-4; - ceres::Solver::Summary summary; - - ceres::Solve(options, &problem, &summary); - - } - }else{ - printf("not enough points in map to associate, map error"); - } - odom = Eigen::Isometry3d::Identity(); - odom.linear() = q_w_curr.toRotationMatrix(); - odom.translation() = t_w_curr; - addPointsToMap(downsampledEdgeCloud,downsampledSurfCloud); - -} - -void OdomEstimationClass::pointAssociateToMap(pcl::PointXYZI const *const pi, pcl::PointXYZI *const po) -{ - Eigen::Vector3d point_curr(pi->x, pi->y, pi->z); - Eigen::Vector3d point_w = q_w_curr * point_curr + t_w_curr; - po->x = point_w.x(); - po->y = point_w.y(); - po->z = point_w.z(); - po->intensity = pi->intensity; - //po->intensity = 1.0; -} - -void OdomEstimationClass::downSamplingToMap(const pcl::PointCloud::Ptr& edge_pc_in, pcl::PointCloud::Ptr& edge_pc_out, const pcl::PointCloud::Ptr& surf_pc_in, pcl::PointCloud::Ptr& surf_pc_out){ - downSizeFilterEdge.setInputCloud(edge_pc_in); - downSizeFilterEdge.filter(*edge_pc_out); - downSizeFilterSurf.setInputCloud(surf_pc_in); - downSizeFilterSurf.filter(*surf_pc_out); -} - -void OdomEstimationClass::addEdgeCostFactor(const pcl::PointCloud::Ptr& pc_in, const pcl::PointCloud::Ptr& map_in, ceres::Problem& problem, ceres::LossFunction *loss_function){ - int corner_num=0; - for (int i = 0; i < (int)pc_in->points.size(); i++) - { - pcl::PointXYZI point_temp; - pointAssociateToMap(&(pc_in->points[i]), &point_temp); - - std::vector pointSearchInd; - std::vector pointSearchSqDis; - kdtreeEdgeMap->nearestKSearch(point_temp, 5, pointSearchInd, pointSearchSqDis); - if (pointSearchSqDis[4] < 1.0) - { - std::vector nearCorners; - Eigen::Vector3d center(0, 0, 0); - for (int j = 0; j < 5; j++) - { - Eigen::Vector3d tmp(map_in->points[pointSearchInd[j]].x, - map_in->points[pointSearchInd[j]].y, - map_in->points[pointSearchInd[j]].z); - center = center + tmp; - nearCorners.push_back(tmp); - } - center = center / 5.0; - - Eigen::Matrix3d covMat = Eigen::Matrix3d::Zero(); - for (int j = 0; j < 5; j++) - { - Eigen::Matrix tmpZeroMean = nearCorners[j] - center; - covMat = covMat + tmpZeroMean * tmpZeroMean.transpose(); - } - - Eigen::SelfAdjointEigenSolver saes(covMat); - - Eigen::Vector3d unit_direction = saes.eigenvectors().col(2); - Eigen::Vector3d curr_point(pc_in->points[i].x, pc_in->points[i].y, pc_in->points[i].z); - if (saes.eigenvalues()[2] > 3 * saes.eigenvalues()[1]) - { - Eigen::Vector3d point_on_line = center; - Eigen::Vector3d point_a, point_b; - point_a = 0.1 * unit_direction + point_on_line; - point_b = -0.1 * unit_direction + point_on_line; - - ceres::CostFunction *cost_function = new EdgeAnalyticCostFunction(curr_point, point_a, point_b); - problem.AddResidualBlock(cost_function, loss_function, parameters); - corner_num++; - } - } - } - if(corner_num<20){ - printf("not enough correct points"); - } - -} - -void OdomEstimationClass::addSurfCostFactor(const pcl::PointCloud::Ptr& pc_in, const pcl::PointCloud::Ptr& map_in, ceres::Problem& problem, ceres::LossFunction *loss_function){ - int surf_num=0; - for (int i = 0; i < (int)pc_in->points.size(); i++) - { - pcl::PointXYZI point_temp; - pointAssociateToMap(&(pc_in->points[i]), &point_temp); - std::vector pointSearchInd; - std::vector pointSearchSqDis; - kdtreeSurfMap->nearestKSearch(point_temp, 5, pointSearchInd, pointSearchSqDis); - - Eigen::Matrix matA0; - Eigen::Matrix matB0 = -1 * Eigen::Matrix::Ones(); - if (pointSearchSqDis[4] < 1.0) - { - - for (int j = 0; j < 5; j++) - { - matA0(j, 0) = map_in->points[pointSearchInd[j]].x; - matA0(j, 1) = map_in->points[pointSearchInd[j]].y; - matA0(j, 2) = map_in->points[pointSearchInd[j]].z; - } - // find the norm of plane - Eigen::Vector3d norm = matA0.colPivHouseholderQr().solve(matB0); - double negative_OA_dot_norm = 1 / norm.norm(); - norm.normalize(); - - bool planeValid = true; - for (int j = 0; j < 5; j++) - { - // if OX * n > 0.2, then plane is not fit well - if (fabs(norm(0) * map_in->points[pointSearchInd[j]].x + - norm(1) * map_in->points[pointSearchInd[j]].y + - norm(2) * map_in->points[pointSearchInd[j]].z + negative_OA_dot_norm) > 0.2) - { - planeValid = false; - break; - } - } - Eigen::Vector3d curr_point(pc_in->points[i].x, pc_in->points[i].y, pc_in->points[i].z); - if (planeValid) - { - ceres::CostFunction *cost_function = new SurfNormAnalyticCostFunction(curr_point, norm, negative_OA_dot_norm); - problem.AddResidualBlock(cost_function, loss_function, parameters); - - surf_num++; - } - } - - } - if(surf_num<20){ - printf("not enough correct points"); - } - -} - -void OdomEstimationClass::addPointsToMap(const pcl::PointCloud::Ptr& downsampledEdgeCloud, const pcl::PointCloud::Ptr& downsampledSurfCloud){ - - for (int i = 0; i < (int)downsampledEdgeCloud->points.size(); i++) - { - pcl::PointXYZI point_temp; - pointAssociateToMap(&downsampledEdgeCloud->points[i], &point_temp); - laserCloudCornerMap->push_back(point_temp); - } - - for (int i = 0; i < (int)downsampledSurfCloud->points.size(); i++) - { - pcl::PointXYZI point_temp; - pointAssociateToMap(&downsampledSurfCloud->points[i], &point_temp); - laserCloudSurfMap->push_back(point_temp); - } - - double x_min = +odom.translation().x()-100; - double y_min = +odom.translation().y()-100; - double z_min = +odom.translation().z()-100; - double x_max = +odom.translation().x()+100; - double y_max = +odom.translation().y()+100; - double z_max = +odom.translation().z()+100; - - //ROS_INFO("size : %f,%f,%f,%f,%f,%f", x_min, y_min, z_min,x_max, y_max, z_max); - cropBoxFilter.setMin(Eigen::Vector4f(x_min, y_min, z_min, 1.0)); - cropBoxFilter.setMax(Eigen::Vector4f(x_max, y_max, z_max, 1.0)); - cropBoxFilter.setNegative(false); - - pcl::PointCloud::Ptr tmpCorner(new pcl::PointCloud()); - pcl::PointCloud::Ptr tmpSurf(new pcl::PointCloud()); - cropBoxFilter.setInputCloud(laserCloudSurfMap); - cropBoxFilter.filter(*tmpSurf); - cropBoxFilter.setInputCloud(laserCloudCornerMap); - cropBoxFilter.filter(*tmpCorner); - - downSizeFilterSurf.setInputCloud(tmpSurf); - downSizeFilterSurf.filter(*laserCloudSurfMap); - downSizeFilterEdge.setInputCloud(tmpCorner); - downSizeFilterEdge.filter(*laserCloudCornerMap); - -} - -void OdomEstimationClass::getMap(pcl::PointCloud::Ptr& laserCloudMap){ - - *laserCloudMap += *laserCloudSurfMap; - *laserCloudMap += *laserCloudCornerMap; -} - -OdomEstimationClass::OdomEstimationClass(){ - -} diff --git a/src/odomEstimationNode.cpp b/src/odomEstimationNode.cpp deleted file mode 100644 index e12ee22..0000000 --- a/src/odomEstimationNode.cpp +++ /dev/null @@ -1,167 +0,0 @@ -// Author of FLOAM: Wang Han -// Email wh200720041@gmail.com -// Homepage https://wanghan.pro - -//c++ lib -#include -#include -#include -#include -#include -#include - -//ros lib -#include -#include -#include -#include -#include - -//pcl lib -#include -#include -#include - -//local lib -#include "lidar.h" -#include "odomEstimationClass.h" - -OdomEstimationClass odomEstimation; -std::mutex mutex_lock; -std::queue pointCloudEdgeBuf; -std::queue pointCloudSurfBuf; -lidar::Lidar lidar_param; - -ros::Publisher pubLaserOdometry; -void velodyneSurfHandler(const sensor_msgs::PointCloud2ConstPtr &laserCloudMsg) -{ - mutex_lock.lock(); - pointCloudSurfBuf.push(laserCloudMsg); - mutex_lock.unlock(); -} -void velodyneEdgeHandler(const sensor_msgs::PointCloud2ConstPtr &laserCloudMsg) -{ - mutex_lock.lock(); - pointCloudEdgeBuf.push(laserCloudMsg); - mutex_lock.unlock(); -} - -bool is_odom_inited = false; -double total_time =0; -int total_frame=0; -void odom_estimation(){ - while(1){ - if(!pointCloudEdgeBuf.empty() && !pointCloudSurfBuf.empty()){ - - //read data - mutex_lock.lock(); - if(!pointCloudSurfBuf.empty() && (pointCloudSurfBuf.front()->header.stamp.toSec()header.stamp.toSec()-0.5*lidar_param.scan_period)){ - pointCloudSurfBuf.pop(); - ROS_WARN_ONCE("time stamp unaligned with extra point cloud, pls check your data --> odom correction"); - mutex_lock.unlock(); - continue; - } - - if(!pointCloudEdgeBuf.empty() && (pointCloudEdgeBuf.front()->header.stamp.toSec()header.stamp.toSec()-0.5*lidar_param.scan_period)){ - pointCloudEdgeBuf.pop(); - ROS_WARN_ONCE("time stamp unaligned with extra point cloud, pls check your data --> odom correction"); - mutex_lock.unlock(); - continue; - } - //if time aligned - - pcl::PointCloud::Ptr pointcloud_surf_in(new pcl::PointCloud()); - pcl::PointCloud::Ptr pointcloud_edge_in(new pcl::PointCloud()); - pcl::fromROSMsg(*pointCloudEdgeBuf.front(), *pointcloud_edge_in); - pcl::fromROSMsg(*pointCloudSurfBuf.front(), *pointcloud_surf_in); - ros::Time pointcloud_time = (pointCloudSurfBuf.front())->header.stamp; - pointCloudEdgeBuf.pop(); - pointCloudSurfBuf.pop(); - mutex_lock.unlock(); - - if(is_odom_inited == false){ - odomEstimation.initMapWithPoints(pointcloud_edge_in, pointcloud_surf_in); - is_odom_inited = true; - ROS_INFO("odom inited"); - }else{ - std::chrono::time_point start, end; - start = std::chrono::system_clock::now(); - odomEstimation.updatePointsToMap(pointcloud_edge_in, pointcloud_surf_in); - end = std::chrono::system_clock::now(); - std::chrono::duration elapsed_seconds = end - start; - total_frame++; - float time_temp = elapsed_seconds.count() * 1000; - total_time+=time_temp; - ROS_INFO("average odom estimation time %f ms \n \n", total_time/total_frame); - } - - - - Eigen::Quaterniond q_current(odomEstimation.odom.rotation()); - //q_current.normalize(); - Eigen::Vector3d t_current = odomEstimation.odom.translation(); - - static tf::TransformBroadcaster br; - tf::Transform transform; - transform.setOrigin( tf::Vector3(t_current.x(), t_current.y(), t_current.z()) ); - tf::Quaternion q(q_current.x(),q_current.y(),q_current.z(),q_current.w()); - transform.setRotation(q); - br.sendTransform(tf::StampedTransform(transform, ros::Time::now(), "map", "base_link")); - - // publish odometry - nav_msgs::Odometry laserOdometry; - laserOdometry.header.frame_id = "map"; - laserOdometry.child_frame_id = "base_link"; - laserOdometry.header.stamp = pointcloud_time; - laserOdometry.pose.pose.orientation.x = q_current.x(); - laserOdometry.pose.pose.orientation.y = q_current.y(); - laserOdometry.pose.pose.orientation.z = q_current.z(); - laserOdometry.pose.pose.orientation.w = q_current.w(); - laserOdometry.pose.pose.position.x = t_current.x(); - laserOdometry.pose.pose.position.y = t_current.y(); - laserOdometry.pose.pose.position.z = t_current.z(); - pubLaserOdometry.publish(laserOdometry); - - } - //sleep 2 ms every time - std::chrono::milliseconds dura(2); - std::this_thread::sleep_for(dura); - } -} - -int main(int argc, char **argv) -{ - ros::init(argc, argv, "main"); - ros::NodeHandle nh; - - int scan_line = 64; - double vertical_angle = 2.0; - double scan_period= 0.1; - double max_dis = 60.0; - double min_dis = 2.0; - double map_resolution = 0.4; - nh.getParam("/scan_period", scan_period); - nh.getParam("/vertical_angle", vertical_angle); - nh.getParam("/max_dis", max_dis); - nh.getParam("/min_dis", min_dis); - nh.getParam("/scan_line", scan_line); - nh.getParam("/map_resolution", map_resolution); - - lidar_param.setScanPeriod(scan_period); - lidar_param.setVerticalAngle(vertical_angle); - lidar_param.setLines(scan_line); - lidar_param.setMaxDistance(max_dis); - lidar_param.setMinDistance(min_dis); - - odomEstimation.init(lidar_param, map_resolution); - ros::Subscriber subEdgeLaserCloud = nh.subscribe("/laser_cloud_edge", 100, velodyneEdgeHandler); - ros::Subscriber subSurfLaserCloud = nh.subscribe("/laser_cloud_surf", 100, velodyneSurfHandler); - - pubLaserOdometry = nh.advertise("/odom", 100); - std::thread odom_estimation_process{odom_estimation}; - - ros::spin(); - - return 0; -} - diff --git a/src/odom_estimation.cpp b/src/odom_estimation.cpp new file mode 100644 index 0000000..0363af0 --- /dev/null +++ b/src/odom_estimation.cpp @@ -0,0 +1,165 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// ros header +#include +#include +#include + +// c++ header +#include +#include + +// ros header +#include +#include + +// local header +#include "floam/odom_estimation_node.hpp" + + +OdomEstimationNode::OdomEstimationNode() + : Node("odom_estimation_node"), is_odom_inited_(false), total_time_(0.0), total_frame_(0) +{ + int scan_line = 64; + double scan_period = 0.1; + double vertical_angle = 2.0; + double max_dist = 60.0; + double min_dist = 3.0; + double map_resolution = 0.4; + + this->declare_parameter("scan_line", scan_line); + this->declare_parameter("scan_period", scan_period); + this->declare_parameter("vertical_angle", vertical_angle); + this->declare_parameter("max_dist", max_dist); + this->declare_parameter("min_dist", min_dist); + this->declare_parameter("map_resolution", map_resolution); + + // load from parameter if provided + scan_line = this->get_parameter("scan_line").get_parameter_value().get(); + scan_period = this->get_parameter("scan_period").get_parameter_value().get(); + vertical_angle = this->get_parameter("vertical_angle").get_parameter_value().get(); + max_dist = this->get_parameter("max_dist").get_parameter_value().get(); + min_dist = this->get_parameter("min_dist").get_parameter_value().get(); + map_resolution = this->get_parameter("map_resolution").get_parameter_value().get(); + + lidar_param_.setScanPeriod(scan_period); + lidar_param_.setVerticalAngle(vertical_angle); + lidar_param_.setLines(scan_line); + lidar_param_.setMaxDistance(max_dist); + lidar_param_.setMinDistance(min_dist); + + odomEstimation_.init(map_resolution); + + subEdgeLaserCloud_ = this->create_subscription( + "laser_cloud_edge", 100, std::bind(&OdomEstimationNode::velodyneEdgeHandler, this, std::placeholders::_1)); + subSurfLaserCloud_ = this->create_subscription( + "laser_cloud_surf", 100, std::bind(&OdomEstimationNode::velodyneSurfHandler, this, std::placeholders::_1)); + + pubLaserOdometry_ = this->create_publisher("odom", 100); + + br_ = std::make_unique(*this); +} + +void OdomEstimationNode::odom_estimation() +{ + while (1) { + if (!pointCloudEdgeBuf_.empty() && !pointCloudSurfBuf_.empty()) { + // read data + mutex_lock_.lock(); + if (!pointCloudSurfBuf_.empty() && (pointCloudSurfBuf_.front()->header.stamp.sec < pointCloudEdgeBuf_.front()->header.stamp.sec - 0.5*lidar_param_.scan_period)) { + pointCloudSurfBuf_.pop(); + RCLCPP_WARN(this->get_logger(), "time stamp unaligned with extra point cloud, pls check your data --> odom correction"); + mutex_lock_.unlock(); + continue; + } + + if (!pointCloudEdgeBuf_.empty() && (pointCloudEdgeBuf_.front()->header.stamp.sec < pointCloudSurfBuf_.front()->header.stamp.sec - 0.5*lidar_param_.scan_period)) { + pointCloudEdgeBuf_.pop(); + RCLCPP_WARN(this->get_logger(), "time stamp unaligned with extra point cloud, pls check your data --> odom correction"); + mutex_lock_.unlock(); + continue; + } + + // if time aligned + pcl::PointCloud::Ptr pointcloud_surf_in(new pcl::PointCloud()); + pcl::PointCloud::Ptr pointcloud_edge_in(new pcl::PointCloud()); + pcl::fromROSMsg(*pointCloudEdgeBuf_.front(), *pointcloud_edge_in); + pcl::fromROSMsg(*pointCloudSurfBuf_.front(), *pointcloud_surf_in); + + rclcpp::Time pointcloud_time = pointCloudSurfBuf_.front()->header.stamp; + pointCloudEdgeBuf_.pop(); + pointCloudSurfBuf_.pop(); + mutex_lock_.unlock(); + + if (is_odom_inited_ == false){ + odomEstimation_.initMapWithPoints(pointcloud_edge_in, pointcloud_surf_in); + is_odom_inited_ = true; + RCLCPP_INFO(this->get_logger(), "odom inited"); + } else { + std::chrono::time_point start, end; + start = std::chrono::system_clock::now(); + odomEstimation_.updatePointsToMap(pointcloud_edge_in, pointcloud_surf_in); + end = std::chrono::system_clock::now(); + std::chrono::duration elapsed_seconds = end - start; + total_frame_++; + float time_temp = elapsed_seconds.count() * 1000; + total_time_ += time_temp; + RCLCPP_INFO(this->get_logger(), "average odom estimation time %f ms\n", total_time_/total_frame_); + } + + Eigen::Quaterniond q_current(odomEstimation_.odom.rotation()); + //q_current.normalize(); + Eigen::Vector3d t_current = odomEstimation_.odom.translation(); + + geometry_msgs::msg::TransformStamped t; + //t.header.stamp = this->get_clock()->now(); + t.header.stamp = pointcloud_time; + t.header.frame_id = "map"; + t.child_frame_id = "base_link"; + + t.transform.translation.x = t_current.x(); + t.transform.translation.y = t_current.y(); + t.transform.translation.z = t_current.z(); + + tf2::Quaternion q(q_current.x(), q_current.y(), q_current.z(), q_current.w()); + t.transform.rotation.x = q.x(); + t.transform.rotation.y = q.y(); + t.transform.rotation.z = q.z(); + t.transform.rotation.w = q.w(); + br_->sendTransform(t); + + // publish odometry + nav_msgs::msg::Odometry laserOdometry; + laserOdometry.header.frame_id = "map"; + laserOdometry.child_frame_id = "base_link"; + laserOdometry.header.stamp = pointcloud_time; + laserOdometry.pose.pose.orientation.x = q_current.x(); + laserOdometry.pose.pose.orientation.y = q_current.y(); + laserOdometry.pose.pose.orientation.z = q_current.z(); + laserOdometry.pose.pose.orientation.w = q_current.w(); + laserOdometry.pose.pose.position.x = t_current.x(); + laserOdometry.pose.pose.position.y = t_current.y(); + laserOdometry.pose.pose.position.z = t_current.z(); + pubLaserOdometry_->publish(laserOdometry); + } + + //sleep 2 ms every time + std::chrono::milliseconds dura(2); + std::this_thread::sleep_for(dura); + } +} + +void OdomEstimationNode::velodyneSurfHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg) +{ + mutex_lock_.lock(); + pointCloudSurfBuf_.push(laserCloudMsg); + mutex_lock_.unlock(); +} + +void OdomEstimationNode::velodyneEdgeHandler(const sensor_msgs::msg::PointCloud2::ConstSharedPtr laserCloudMsg) +{ + mutex_lock_.lock(); + pointCloudEdgeBuf_.push(laserCloudMsg); + mutex_lock_.unlock(); +} diff --git a/src/odom_estimation_class.cpp b/src/odom_estimation_class.cpp new file mode 100644 index 0000000..ed8a779 --- /dev/null +++ b/src/odom_estimation_class.cpp @@ -0,0 +1,258 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// local header +#include "floam/odom_estimation_class.hpp" + + +void OdomEstimationClass::init(double map_resolution) +{ + // init local map + laserCloudCornerMap = pcl::PointCloud::Ptr(new pcl::PointCloud()); + laserCloudSurfMap = pcl::PointCloud::Ptr(new pcl::PointCloud()); + + // downsampling size + downSizeFilterEdge.setLeafSize(map_resolution, map_resolution, map_resolution); + downSizeFilterSurf.setLeafSize(map_resolution * 2, map_resolution * 2, map_resolution * 2); + + // kd-tree + kdtreeEdgeMap = pcl::KdTreeFLANN::Ptr(new pcl::KdTreeFLANN()); + kdtreeSurfMap = pcl::KdTreeFLANN::Ptr(new pcl::KdTreeFLANN()); + + odom = Eigen::Isometry3d::Identity(); + last_odom = Eigen::Isometry3d::Identity(); + optimization_count = 2; +} + +void OdomEstimationClass::initMapWithPoints(const pcl::PointCloud::Ptr& edge_in, + const pcl::PointCloud::Ptr& surf_in) +{ + *laserCloudCornerMap += *edge_in; + *laserCloudSurfMap += *surf_in; + optimization_count = 12; +} + +void OdomEstimationClass::updatePointsToMap(const pcl::PointCloud::Ptr& edge_in, + const pcl::PointCloud::Ptr& surf_in) +{ + if (optimization_count > 2) { + optimization_count--; + } + + Eigen::Isometry3d odom_prediction = odom * (last_odom.inverse() * odom); + last_odom = odom; + odom = odom_prediction; + + q_w_curr = Eigen::Quaterniond(odom.rotation()); + t_w_curr = odom.translation(); + + pcl::PointCloud::Ptr downsampledEdgeCloud(new pcl::PointCloud()); + pcl::PointCloud::Ptr downsampledSurfCloud(new pcl::PointCloud()); + downSamplingToMap(edge_in,downsampledEdgeCloud,surf_in,downsampledSurfCloud); + // printf("point nyum%d,%d",(int)downsampledEdgeCloud->points.size(), (int)downsampledSurfCloud->points.size()); + + if (laserCloudCornerMap->points.size() > 10 && laserCloudSurfMap->points.size() > 50) { + kdtreeEdgeMap->setInputCloud(laserCloudCornerMap); + kdtreeSurfMap->setInputCloud(laserCloudSurfMap); + + for (int iterCount = 0; iterCount < optimization_count; iterCount++) { + ceres::LossFunction *loss_function = new ceres::HuberLoss(0.1); + ceres::Problem::Options problem_options; + ceres::Problem problem(problem_options); + + problem.AddParameterBlock(parameters, 7, new PoseSE3Parameterization()); + + addEdgeCostFactor(downsampledEdgeCloud,laserCloudCornerMap,problem,loss_function); + addSurfCostFactor(downsampledSurfCloud,laserCloudSurfMap,problem,loss_function); + + ceres::Solver::Options options; + options.linear_solver_type = ceres::DENSE_QR; + options.max_num_iterations = 4; + options.minimizer_progress_to_stdout = false; + options.check_gradients = false; + options.gradient_check_relative_precision = 1e-4; + ceres::Solver::Summary summary; + + ceres::Solve(options, &problem, &summary); + } + } else { + printf("not enough points in map to associate, map error"); + } + + odom = Eigen::Isometry3d::Identity(); + odom.linear() = q_w_curr.toRotationMatrix(); + odom.translation() = t_w_curr; + addPointsToMap(downsampledEdgeCloud,downsampledSurfCloud); +} + +void OdomEstimationClass::getMap(pcl::PointCloud::Ptr& laserCloudMap) +{ + *laserCloudMap += *laserCloudSurfMap; + *laserCloudMap += *laserCloudCornerMap; +} + +void OdomEstimationClass::addEdgeCostFactor(const pcl::PointCloud::Ptr& pc_in, + const pcl::PointCloud::Ptr& map_in, ceres::Problem& problem, ceres::LossFunction *loss_function) +{ + int corner_num = 0; + for (int i = 0; i < (int)pc_in->points.size(); i++) { + pcl::PointXYZI point_temp; + pointAssociateToMap(&(pc_in->points[i]), &point_temp); + + std::vector pointSearchInd; + std::vector pointSearchSqDis; + kdtreeEdgeMap->nearestKSearch(point_temp, 5, pointSearchInd, pointSearchSqDis); + + if (pointSearchSqDis[4] < 1.0) { + std::vector nearCorners; + Eigen::Vector3d center(0, 0, 0); + for (int j = 0; j < 5; j++) { + Eigen::Vector3d tmp(map_in->points[pointSearchInd[j]].x, + map_in->points[pointSearchInd[j]].y, + map_in->points[pointSearchInd[j]].z); + center = center + tmp; + nearCorners.push_back(tmp); + } + center = center / 5.0; + + Eigen::Matrix3d covMat = Eigen::Matrix3d::Zero(); + for (int j = 0; j < 5; j++) { + Eigen::Matrix tmpZeroMean = nearCorners[j] - center; + covMat = covMat + tmpZeroMean * tmpZeroMean.transpose(); + } + + Eigen::SelfAdjointEigenSolver saes(covMat); + + Eigen::Vector3d unit_direction = saes.eigenvectors().col(2); + Eigen::Vector3d curr_point(pc_in->points[i].x, pc_in->points[i].y, pc_in->points[i].z); + + if (saes.eigenvalues()[2] > 3 * saes.eigenvalues()[1]) { + Eigen::Vector3d point_on_line = center; + Eigen::Vector3d point_a, point_b; + point_a = 0.1 * unit_direction + point_on_line; + point_b = -0.1 * unit_direction + point_on_line; + + ceres::CostFunction *cost_function = new EdgeAnalyticCostFunction(curr_point, point_a, point_b); + problem.AddResidualBlock(cost_function, loss_function, parameters); + corner_num++; + } + } + } + + if (corner_num < 20) { + printf("not enough correct points"); + } +} + +void OdomEstimationClass::addSurfCostFactor(const pcl::PointCloud::Ptr& pc_in, + const pcl::PointCloud::Ptr& map_in, ceres::Problem& problem, ceres::LossFunction *loss_function) +{ + int surf_num=0; + for (int i = 0; i < (int)pc_in->points.size(); i++) { + pcl::PointXYZI point_temp; + pointAssociateToMap(&(pc_in->points[i]), &point_temp); + std::vector pointSearchInd; + std::vector pointSearchSqDis; + kdtreeSurfMap->nearestKSearch(point_temp, 5, pointSearchInd, pointSearchSqDis); + + Eigen::Matrix matA0; + Eigen::Matrix matB0 = -1 * Eigen::Matrix::Ones(); + + if (pointSearchSqDis[4] < 1.0) { + for (int j = 0; j < 5; j++) { + matA0(j, 0) = map_in->points[pointSearchInd[j]].x; + matA0(j, 1) = map_in->points[pointSearchInd[j]].y; + matA0(j, 2) = map_in->points[pointSearchInd[j]].z; + } + // find the norm of plane + Eigen::Vector3d norm = matA0.colPivHouseholderQr().solve(matB0); + double negative_OA_dot_norm = 1 / norm.norm(); + norm.normalize(); + + bool planeValid = true; + for (int j = 0; j < 5; j++) { + // if OX * n > 0.2, then plane is not fit well + if (fabs(norm(0) * map_in->points[pointSearchInd[j]].x + + norm(1) * map_in->points[pointSearchInd[j]].y + + norm(2) * map_in->points[pointSearchInd[j]].z + + negative_OA_dot_norm) > 0.2) { + planeValid = false; + break; + } + } + + Eigen::Vector3d curr_point(pc_in->points[i].x, pc_in->points[i].y, pc_in->points[i].z); + + if (planeValid) { + ceres::CostFunction *cost_function = new SurfNormAnalyticCostFunction(curr_point, norm, negative_OA_dot_norm); + problem.AddResidualBlock(cost_function, loss_function, parameters); + surf_num++; + } + } + } + + if (surf_num < 20) { + printf("not enough correct points"); + } + +} + +void OdomEstimationClass::addPointsToMap(const pcl::PointCloud::Ptr& downsampledEdgeCloud, + const pcl::PointCloud::Ptr& downsampledSurfCloud) +{ + for (int i = 0; i < (int)downsampledEdgeCloud->points.size(); i++) { + pcl::PointXYZI point_temp; + pointAssociateToMap(&downsampledEdgeCloud->points[i], &point_temp); + laserCloudCornerMap->push_back(point_temp); + } + + for (int i = 0; i < (int)downsampledSurfCloud->points.size(); i++) { + pcl::PointXYZI point_temp; + pointAssociateToMap(&downsampledSurfCloud->points[i], &point_temp); + laserCloudSurfMap->push_back(point_temp); + } + + double x_min = +odom.translation().x()-100; + double y_min = +odom.translation().y()-100; + double z_min = +odom.translation().z()-100; + double x_max = +odom.translation().x()+100; + double y_max = +odom.translation().y()+100; + double z_max = +odom.translation().z()+100; + + // printf("size : %f,%f,%f,%f,%f,%f", x_min, y_min, z_min,x_max, y_max, z_max); + cropBoxFilter.setMin(Eigen::Vector4f(x_min, y_min, z_min, 1.0)); + cropBoxFilter.setMax(Eigen::Vector4f(x_max, y_max, z_max, 1.0)); + cropBoxFilter.setNegative(false); + + pcl::PointCloud::Ptr tmpCorner(new pcl::PointCloud()); + pcl::PointCloud::Ptr tmpSurf(new pcl::PointCloud()); + cropBoxFilter.setInputCloud(laserCloudSurfMap); + cropBoxFilter.filter(*tmpSurf); + cropBoxFilter.setInputCloud(laserCloudCornerMap); + cropBoxFilter.filter(*tmpCorner); + + downSizeFilterSurf.setInputCloud(tmpSurf); + downSizeFilterSurf.filter(*laserCloudSurfMap); + downSizeFilterEdge.setInputCloud(tmpCorner); + downSizeFilterEdge.filter(*laserCloudCornerMap); +} + +void OdomEstimationClass::pointAssociateToMap(pcl::PointXYZI const *const pi, pcl::PointXYZI *const po) +{ + Eigen::Vector3d point_curr(pi->x, pi->y, pi->z); + Eigen::Vector3d point_w = q_w_curr * point_curr + t_w_curr; + po->x = point_w.x(); + po->y = point_w.y(); + po->z = point_w.z(); + po->intensity = pi->intensity; +} + +void OdomEstimationClass::downSamplingToMap(const pcl::PointCloud::Ptr& edge_pc_in, + pcl::PointCloud::Ptr& edge_pc_out, const pcl::PointCloud::Ptr& surf_pc_in, + pcl::PointCloud::Ptr& surf_pc_out) +{ + downSizeFilterEdge.setInputCloud(edge_pc_in); + downSizeFilterEdge.filter(*edge_pc_out); + downSizeFilterSurf.setInputCloud(surf_pc_in); + downSizeFilterSurf.filter(*surf_pc_out); +} diff --git a/src/odom_estimation_node.cpp b/src/odom_estimation_node.cpp new file mode 100644 index 0000000..fe1f09e --- /dev/null +++ b/src/odom_estimation_node.cpp @@ -0,0 +1,23 @@ +// Author of FLOAM: Wang Han +// ROS2 Migration: Yi-Chen Zhang + +// ros header +#include + +// c++ header +#include + +// local header +#include "floam/odom_estimation_node.hpp" + + +int main(int argc, char* argv[]) +{ + rclcpp::init(argc, argv); + auto oen_ptr {std::make_shared()}; + std::thread odom_estimation_process(&OdomEstimationNode::odom_estimation, oen_ptr); + rclcpp::spin(oen_ptr); + rclcpp::shutdown(); + + return 0; +}