laserMapping.cpp
Go to the documentation of this file.
00001 #include <math.h>
00002 #include <time.h>
00003 #include <stdio.h>
00004 #include <stdlib.h>
00005 #include <ros/ros.h>
00006 
00007 #include <nav_msgs/Odometry.h>
00008 #include <sensor_msgs/Imu.h>
00009 #include <sensor_msgs/PointCloud2.h>
00010 
00011 #include <tf/transform_datatypes.h>
00012 #include <tf/transform_broadcaster.h>
00013 
00014 #include <opencv/cv.h>
00015 #include <opencv2/highgui/highgui.hpp>
00016 
00017 #include <pcl/ros/conversions.h>
00018 #include <pcl/point_cloud.h>
00019 #include <pcl/point_types.h>
00020 #include <pcl/filters/voxel_grid.h>
00021 #include <pcl/kdtree/kdtree_flann.h>
00022 
00023 const double PI = 3.1415926;
00024 const double rad2deg = 180 / PI;
00025 const double deg2rad = PI / 180;
00026 
00027 double timeLaserCloudLast;
00028 double timeLaserOdometry;
00029 
00030 bool newLaserCloudLast = false;
00031 bool newLaserOdometry = false;
00032 
00033 const int laserCloudCenWidth = 5;
00034 const int laserCloudCenHeight = 5;
00035 const int laserCloudCenDepth = 5;
00036 const int laserCloudWidth = 11;
00037 const int laserCloudHeight = 11;
00038 const int laserCloudDepth = 11;
00039 const int laserCloudNum = laserCloudWidth * laserCloudHeight * laserCloudDepth;
00040 
00041 int laserCloudValidInd[27];
00042 int laserCloudSurroundInd[27];
00043 
00044 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudCorner(new pcl::PointCloud<pcl::PointXYZHSV>());
00045 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudSurf(new pcl::PointCloud<pcl::PointXYZHSV>());
00046 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudCorner2(new pcl::PointCloud<pcl::PointXYZHSV>());
00047 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudSurf2(new pcl::PointCloud<pcl::PointXYZHSV>());
00048 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudLast(new pcl::PointCloud<pcl::PointXYZHSV>());
00049 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudOri(new pcl::PointCloud<pcl::PointXYZHSV>());
00050 //pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudSel(new pcl::PointCloud<pcl::PointXYZHSV>());
00051 //pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudCorr(new pcl::PointCloud<pcl::PointXYZHSV>());
00052 //pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudProj(new pcl::PointCloud<pcl::PointXYZHSV>());
00053 pcl::PointCloud<pcl::PointXYZHSV>::Ptr coeffSel(new pcl::PointCloud<pcl::PointXYZHSV>());
00054 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudFromMap(new pcl::PointCloud<pcl::PointXYZHSV>());
00055 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudCornerFromMap(new pcl::PointCloud<pcl::PointXYZHSV>());
00056 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudSurfFromMap(new pcl::PointCloud<pcl::PointXYZHSV>());
00057 pcl::PointCloud<pcl::PointXYZI>::Ptr laserCloudSurround(new pcl::PointCloud<pcl::PointXYZI>());
00058 pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudArray[laserCloudNum];
00059 
00060 pcl::KdTreeFLANN<pcl::PointXYZHSV>::Ptr kdtreeCornerFromMap(new pcl::KdTreeFLANN<pcl::PointXYZHSV>());
00061 pcl::KdTreeFLANN<pcl::PointXYZHSV>::Ptr kdtreeSurfFromMap(new pcl::KdTreeFLANN<pcl::PointXYZHSV>());
00062 
00063 float transformSum[6] = {0};
00064 float transformIncre[6] = {0};
00065 float transformTobeMapped[6] = {0};
00066 float transformBefMapped[6] = {0};
00067 float transformAftMapped[6] = {0};
00068 
00069 void transformAssociateToMap()
00070 {
00071   float x1 = cos(transformSum[1]) * (transformBefMapped[3] - transformSum[3]) 
00072            - sin(transformSum[1]) * (transformBefMapped[5] - transformSum[5]);
00073   float y1 = transformBefMapped[4] - transformSum[4];
00074   float z1 = sin(transformSum[1]) * (transformBefMapped[3] - transformSum[3]) 
00075            + cos(transformSum[1]) * (transformBefMapped[5] - transformSum[5]);
00076 
00077   float x2 = x1;
00078   float y2 = cos(transformSum[0]) * y1 + sin(transformSum[0]) * z1;
00079   float z2 = -sin(transformSum[0]) * y1 + cos(transformSum[0]) * z1;
00080 
00081   transformIncre[3] = cos(transformSum[2]) * x2 + sin(transformSum[2]) * y2;
00082   transformIncre[4] = -sin(transformSum[2]) * x2 + cos(transformSum[2]) * y2;
00083   transformIncre[5] = z2;
00084 
00085   transformIncre[0] = transformBefMapped[0] - transformSum[0];
00086   transformIncre[1] = transformBefMapped[1] - transformSum[1];
00087   transformIncre[2] = transformBefMapped[2] - transformSum[2];
00088 
00089   transformTobeMapped[0] = transformAftMapped[0] - transformIncre[0];
00090   transformTobeMapped[1] = transformAftMapped[1] - transformIncre[1];
00091   transformTobeMapped[2] = transformAftMapped[2] - transformIncre[2];
00092 
00093   x1 = cos(transformTobeMapped[2]) * transformIncre[3] - sin(transformTobeMapped[2]) * transformIncre[4];
00094   y1 = sin(transformTobeMapped[2]) * transformIncre[3] + cos(transformTobeMapped[2]) * transformIncre[4];
00095   z1 = transformIncre[5];
00096 
00097   x2 = x1;
00098   y2 = cos(transformTobeMapped[0]) * y1 - sin(transformTobeMapped[0]) * z1;
00099   z2 = sin(transformTobeMapped[0]) * y1 + cos(transformTobeMapped[0]) * z1;
00100 
00101   transformTobeMapped[3] = transformAftMapped[3] 
00102                          - (cos(transformTobeMapped[1]) * x2 + sin(transformTobeMapped[1]) * z2);
00103   transformTobeMapped[4] = transformAftMapped[4] - y2;
00104   transformTobeMapped[5] = transformAftMapped[5] 
00105                          - (-sin(transformTobeMapped[1]) * x2 + cos(transformTobeMapped[1]) * z2);
00106 }
00107 
00108 void transformUpdate()
00109 {
00110   //for (int i = 0; i < 3; i++) {
00111   //  transformTobeMapped[i] = (1 - 0.1) * transformTobeMapped[i] + 0.1 * transformSum[i];
00112   //}
00113 
00114   for (int i = 0; i < 6; i++) {
00115     transformBefMapped[i] = transformSum[i];
00116     transformAftMapped[i] = transformTobeMapped[i];
00117   }
00118 }
00119 
00120 void pointAssociateToMap(pcl::PointXYZHSV *pi, pcl::PointXYZHSV *po)
00121 {
00122   float x1 = cos(transformTobeMapped[2]) * pi->x
00123            - sin(transformTobeMapped[2]) * pi->y;
00124   float y1 = sin(transformTobeMapped[2]) * pi->x
00125            + cos(transformTobeMapped[2]) * pi->y;
00126   float z1 = pi->z;
00127 
00128   float x2 = x1;
00129   float y2 = cos(transformTobeMapped[0]) * y1 - sin(transformTobeMapped[0]) * z1;
00130   float z2 = sin(transformTobeMapped[0]) * y1 + cos(transformTobeMapped[0]) * z1;
00131 
00132   po->x = cos(transformTobeMapped[1]) * x2 + sin(transformTobeMapped[1]) * z2
00133         + transformTobeMapped[3];
00134   po->y = y2 + transformTobeMapped[4];
00135   po->z = -sin(transformTobeMapped[1]) * x2 + cos(transformTobeMapped[1]) * z2
00136         + transformTobeMapped[5];
00137   po->h = pi->h;
00138   po->s = pi->s;
00139   po->v = pi->v;
00140 }
00141 
00142 void laserCloudLastHandler(const sensor_msgs::PointCloud2ConstPtr& laserCloudLast2)
00143 {
00144   timeLaserCloudLast = laserCloudLast2->header.stamp.toSec();
00145 
00146   laserCloudLast->clear();
00147   pcl::fromROSMsg(*laserCloudLast2, *laserCloudLast);
00148 
00149   newLaserCloudLast = true;
00150 }
00151 
00152 void laserOdometryHandler(const nav_msgs::Odometry::ConstPtr& laserOdometry)
00153 {
00154   timeLaserOdometry = laserOdometry->header.stamp.toSec();
00155 
00156   double roll, pitch, yaw;
00157   geometry_msgs::Quaternion geoQuat = laserOdometry->pose.pose.orientation;
00158   tf::Matrix3x3(tf::Quaternion(geoQuat.z, -geoQuat.x, -geoQuat.y, geoQuat.w)).getRPY(roll, pitch, yaw);
00159 
00160   transformSum[0] = -pitch;
00161   transformSum[1] = -yaw;
00162   transformSum[2] = roll;
00163 
00164   transformSum[3] = laserOdometry->pose.pose.position.x;
00165   transformSum[4] = laserOdometry->pose.pose.position.y;
00166   transformSum[5] = laserOdometry->pose.pose.position.z;
00167 
00168   newLaserOdometry = true;
00169 }
00170 
00171 int main(int argc, char** argv)
00172 {
00173   ros::init(argc, argv, "laserMapping");
00174   ros::NodeHandle nh;
00175 
00176   ros::Subscriber subLaserCloudLast2 = nh.subscribe<sensor_msgs::PointCloud2> 
00177                                        ("/laser_cloud_last_2", 2, laserCloudLastHandler);
00178 
00179   ros::Subscriber subLaserOdometry = nh.subscribe<nav_msgs::Odometry> 
00180                                      ("/cam_to_init", 5, laserOdometryHandler);
00181 
00182   ros::Publisher pubLaserCloudSurround = nh.advertise<sensor_msgs::PointCloud2> 
00183                                          ("/laser_cloud_surround", 1);
00184 
00185   //ros::Publisher pub1 = nh.advertise<sensor_msgs::PointCloud2> ("/pc1", 1);
00186 
00187   //ros::Publisher pub2 = nh.advertise<sensor_msgs::PointCloud2> ("/pc2", 1);
00188 
00189   //ros::Publisher pub3 = nh.advertise<sensor_msgs::PointCloud2> ("/pc3", 1);
00190 
00191   //ros::Publisher pub4 = nh.advertise<sensor_msgs::PointCloud2> ("/pc4", 1);
00192 
00193   ros::Publisher pubOdomBefMapped = nh.advertise<nav_msgs::Odometry> ("/bef_mapped_to_init_2", 5);
00194   nav_msgs::Odometry odomBefMapped;
00195   odomBefMapped.header.frame_id = "/camera_init_2";
00196   odomBefMapped.child_frame_id = "/bef_mapped";
00197 
00198   ros::Publisher pubOdomAftMapped = nh.advertise<nav_msgs::Odometry> ("/aft_mapped_to_init_2", 5);
00199   nav_msgs::Odometry odomAftMapped;
00200   odomAftMapped.header.frame_id = "/camera_init_2";
00201   odomAftMapped.child_frame_id = "/aft_mapped";
00202 
00203   std::vector<int> pointSearchInd;
00204   std::vector<float> pointSearchSqDis;
00205 
00206   pcl::PointXYZHSV pointOri, pointSel, pointProj, coeff;
00207   pcl::PointXYZI pointSurround;
00208 
00209   cv::Mat matA0(5, 3, CV_32F, cv::Scalar::all(0));
00210   cv::Mat matB0(5, 1, CV_32F, cv::Scalar::all(-1));
00211   cv::Mat matX0(3, 1, CV_32F, cv::Scalar::all(0));
00212 
00213   cv::Mat matA1(3, 3, CV_32F, cv::Scalar::all(0));
00214   cv::Mat matD1(1, 3, CV_32F, cv::Scalar::all(0));
00215   cv::Mat matV1(3, 3, CV_32F, cv::Scalar::all(0));
00216 
00217   for (int i = 0; i < laserCloudNum; i++) {
00218     laserCloudArray[i].reset(new pcl::PointCloud<pcl::PointXYZHSV>());
00219   }
00220 
00221   bool status = ros::ok();
00222   while (status) {
00223     ros::spinOnce();
00224 
00225     if (newLaserCloudLast && newLaserOdometry && fabs(timeLaserCloudLast - timeLaserOdometry) < 0.005) {
00226       newLaserCloudLast = false;
00227       newLaserOdometry = false;
00228 
00229       transformAssociateToMap();
00230 
00231       pcl::PointXYZHSV pointOnZAxis;
00232       pointOnZAxis.x = 0.0;
00233       pointOnZAxis.y = 0.0;
00234       pointOnZAxis.z = 10.0;
00235       pointAssociateToMap(&pointOnZAxis, &pointOnZAxis);        
00236 
00237       int centerCubeI = int((transformTobeMapped[3] + 10.0) / 20.0) + laserCloudCenWidth;
00238       int centerCubeJ = int((transformTobeMapped[4] + 10.0) / 20.0) + laserCloudCenHeight;
00239       int centerCubeK = int((transformTobeMapped[5] + 10.0) / 20.0) + laserCloudCenDepth;
00240       if (centerCubeI < 0 || centerCubeI >= laserCloudWidth || 
00241           centerCubeJ < 0 || centerCubeJ >= laserCloudHeight || 
00242           centerCubeK < 0 || centerCubeK >= laserCloudDepth) {
00243 
00244         transformUpdate();
00245         continue;
00246       }
00247 
00248       int laserCloudValidNum = 0;
00249       int laserCloudSurroundNum = 0;
00250       for (int i = centerCubeI - 1; i <= centerCubeI + 1; i++) {
00251         for (int j = centerCubeJ - 1; j <= centerCubeJ + 1; j++) {
00252           for (int k = centerCubeK - 1; k <= centerCubeK + 1; k++) {
00253             if (i >= 0 && i < laserCloudWidth && 
00254                 j >= 0 && j < laserCloudHeight && 
00255                 k >= 0 && k < laserCloudDepth) {
00256               
00257               float centerX = 20.0 * (i - laserCloudCenWidth);
00258               float centerY = 20.0 * (j - laserCloudCenHeight);
00259               float centerZ = 20.0 * (k - laserCloudCenDepth);
00260 
00261               bool isInLaserFOV = false;
00262               for (int ii = -1; ii <= 1; ii += 2) {
00263                 for (int jj = -1; jj <= 1; jj += 2) {
00264                   for (int kk = -1; kk <= 1; kk += 2) {
00265                     float cornerX = centerX + 10.0 * ii;
00266                     float cornerY = centerY + 10.0 * jj;
00267                     float cornerZ = centerZ + 10.0 * kk;
00268 
00269                     float check = 100.0 
00270                                 + (transformTobeMapped[3] - cornerX) * (transformTobeMapped[3] - cornerX) 
00271                                 + (transformTobeMapped[4] - cornerY) * (transformTobeMapped[4] - cornerY)
00272                                 + (transformTobeMapped[5] - cornerZ) * (transformTobeMapped[5] - cornerZ)
00273                                 - (pointOnZAxis.x - cornerX) * (pointOnZAxis.x - cornerX) 
00274                                 - (pointOnZAxis.y - cornerY) * (pointOnZAxis.y - cornerY)
00275                                 - (pointOnZAxis.z - cornerZ) * (pointOnZAxis.z - cornerZ);
00276 
00277                     if (check > 0) {
00278                       isInLaserFOV = true;
00279                     }
00280                   }
00281                 }
00282               }
00283 
00284               if (isInLaserFOV) {
00285                 laserCloudValidInd[laserCloudValidNum] = i + laserCloudWidth * j 
00286                                                        + laserCloudWidth * laserCloudHeight * k;
00287                 laserCloudValidNum++;
00288               }
00289               laserCloudSurroundInd[laserCloudSurroundNum] = i + laserCloudWidth * j 
00290                                                            + laserCloudWidth * laserCloudHeight * k;
00291               laserCloudSurroundNum++;
00292             }
00293           }
00294         }
00295       }
00296 
00297       laserCloudFromMap->clear();
00298       for (int i = 0; i < laserCloudValidNum; i++) {
00299         *laserCloudFromMap += *laserCloudArray[laserCloudValidInd[i]];
00300       }
00301       int laserCloudFromMapNum = laserCloudFromMap->points.size();
00302 
00303       laserCloudCornerFromMap->clear();
00304       laserCloudSurfFromMap->clear();
00305       for (int i = 0; i < laserCloudFromMapNum; i++) {
00306         if (fabs(laserCloudFromMap->points[i].v - 2) < 0.005 || 
00307             fabs(laserCloudFromMap->points[i].v - 1) < 0.005) {
00308           laserCloudCornerFromMap->push_back(laserCloudFromMap->points[i]);
00309         } else {
00310           laserCloudSurfFromMap->push_back(laserCloudFromMap->points[i]);
00311         }
00312       }
00313 
00314       laserCloudCorner->clear();
00315       laserCloudSurf->clear();
00316       int laserCloudLastNum = laserCloudLast->points.size();
00317       for (int i = 0; i < laserCloudLastNum; i++) {
00318         if (fabs(laserCloudLast->points[i].v - 2) < 0.005 || 
00319             fabs(laserCloudLast->points[i].v - 1) < 0.005) {
00320           laserCloudCorner->push_back(laserCloudLast->points[i]);
00321         } else {
00322           laserCloudSurf->push_back(laserCloudLast->points[i]);
00323         }
00324       }
00325 
00326       laserCloudCorner2->clear();
00327       pcl::VoxelGrid<pcl::PointXYZHSV> downSizeFilter;
00328       downSizeFilter.setInputCloud(laserCloudCorner);
00329       downSizeFilter.setLeafSize(0.05, 0.05, 0.05);
00330       downSizeFilter.filter(*laserCloudCorner2);
00331 
00332       laserCloudSurf2->clear();
00333       downSizeFilter.setInputCloud(laserCloudSurf);
00334       downSizeFilter.setLeafSize(0.1, 0.1, 0.1);
00335       downSizeFilter.filter(*laserCloudSurf2);
00336 
00337       laserCloudLast->clear();
00338       *laserCloudLast = *laserCloudCorner2 + *laserCloudSurf2;
00339       laserCloudLastNum = laserCloudLast->points.size();
00340 
00341       if (laserCloudSurfFromMap->points.size() > 500) {
00342 
00343         kdtreeCornerFromMap->setInputCloud(laserCloudCornerFromMap);
00344         kdtreeSurfFromMap->setInputCloud(laserCloudSurfFromMap);
00345 
00346         for (int iterCount = 0; iterCount < 10; iterCount++) {
00347           laserCloudOri->clear();
00348           //laserCloudSel->clear();
00349           //laserCloudCorr->clear();
00350           //laserCloudProj->clear();
00351           coeffSel->clear();
00352 
00353           for (int i = 0; i < laserCloudLastNum; i++) {
00354             if (fabs(laserCloudLast->points[i].x > 0.3) || fabs(laserCloudLast->points[i].y > 0.3) ||
00355                 fabs(laserCloudLast->points[i].z > 0.3)) {
00356 
00357               pointOri = laserCloudLast->points[i];
00358               pointAssociateToMap(&pointOri, &pointSel); 
00359               if (fabs(pointOri.v) < 0.05 || fabs(pointOri.v + 1) < 0.05) {
00360 
00361                 kdtreeSurfFromMap->nearestKSearch(pointSel, 5, pointSearchInd, pointSearchSqDis);
00362 
00363                 if (pointSearchSqDis[4] < 1.0) {
00364                   for (int j = 0; j < 5; j++) {
00365                     matA0.at<float>(j, 0) = laserCloudSurfFromMap->points[pointSearchInd[j]].x;
00366                     matA0.at<float>(j, 1) = laserCloudSurfFromMap->points[pointSearchInd[j]].y;
00367                     matA0.at<float>(j, 2) = laserCloudSurfFromMap->points[pointSearchInd[j]].z;
00368                   }
00369                   cv::solve(matA0, matB0, matX0, cv::DECOMP_QR);
00370 
00371                   float pa = matX0.at<float>(0, 0);
00372                   float pb = matX0.at<float>(1, 0);
00373                   float pc = matX0.at<float>(2, 0);
00374                   float pd = 1;
00375  
00376                   float ps = sqrt(pa * pa + pb * pb + pc * pc);
00377                   pa /= ps;
00378                   pb /= ps;
00379                   pc /= ps;
00380                   pd /= ps;
00381 
00382                   bool planeValid = true;
00383                   for (int j = 0; j < 5; j++) {
00384                     if (fabs(pa * laserCloudSurfFromMap->points[pointSearchInd[j]].x +
00385                         pb * laserCloudSurfFromMap->points[pointSearchInd[j]].y +
00386                         pc * laserCloudSurfFromMap->points[pointSearchInd[j]].z + pd) > 0.05) {
00387                       planeValid = false;
00388                       break;
00389                     }
00390                   }
00391 
00392                   if (planeValid) {
00393                     float pd2 = pa * pointSel.x + pb * pointSel.y + pc * pointSel.z + pd;
00394 
00395                     pointProj = pointSel;
00396                     pointProj.x -= pa * pd2;
00397                     pointProj.y -= pb * pd2;
00398                     pointProj.z -= pc * pd2;
00399 
00400                     float s = 1;
00401                     if (iterCount >= 6) {
00402                       s = 1 - 8 * fabs(pd2) / sqrt(sqrt(pointSel.x * pointSel.x
00403                         + pointSel.y * pointSel.y + pointSel.z * pointSel.z));
00404                     }
00405 
00406                     coeff.x = s * pa;
00407                     coeff.y = s * pb;
00408                     coeff.z = s * pc;
00409                     coeff.h = s * pd2;
00410 
00411                     if (s > 0.2) {
00412                       laserCloudOri->push_back(pointOri);
00413                       //laserCloudSel->push_back(pointSel);
00414                       //laserCloudProj->push_back(pointProj);
00415                       //laserCloudCorr->push_back(laserCloudSurfFromMap->points[pointSearchInd[0]]);
00416                       //laserCloudCorr->push_back(laserCloudSurfFromMap->points[pointSearchInd[1]]);
00417                       //laserCloudCorr->push_back(laserCloudSurfFromMap->points[pointSearchInd[2]]);
00418                       //laserCloudCorr->push_back(laserCloudSurfFromMap->points[pointSearchInd[3]]);
00419                       //laserCloudCorr->push_back(laserCloudSurfFromMap->points[pointSearchInd[4]]);
00420                       coeffSel->push_back(coeff);
00421                     }
00422                   }
00423                 }
00424               } else {
00425 
00426                 kdtreeCornerFromMap->nearestKSearch(pointSel, 5, pointSearchInd, pointSearchSqDis);
00427                 
00428                 if (pointSearchSqDis[4] < 1.0) {
00429                   float cx = 0;
00430                   float cy = 0; 
00431                   float cz = 0;
00432                   for (int j = 0; j < 5; j++) {
00433                     cx += laserCloudCornerFromMap->points[pointSearchInd[j]].x;
00434                     cy += laserCloudCornerFromMap->points[pointSearchInd[j]].y;
00435                     cz += laserCloudCornerFromMap->points[pointSearchInd[j]].z;
00436                   }
00437                   cx /= 5;
00438                   cy /= 5; 
00439                   cz /= 5;
00440 
00441                   float a11 = 0;
00442                   float a12 = 0; 
00443                   float a13 = 0;
00444                   float a22 = 0;
00445                   float a23 = 0; 
00446                   float a33 = 0;
00447                   for (int j = 0; j < 5; j++) {
00448                     float ax = laserCloudCornerFromMap->points[pointSearchInd[j]].x - cx;
00449                     float ay = laserCloudCornerFromMap->points[pointSearchInd[j]].y - cy;
00450                     float az = laserCloudCornerFromMap->points[pointSearchInd[j]].z - cz;
00451 
00452                     a11 += ax * ax;
00453                     a12 += ax * ay;
00454                     a13 += ax * az;
00455                     a22 += ay * ay;
00456                     a23 += ay * az;
00457                     a33 += az * az;
00458                   }
00459                   a11 /= 5;
00460                   a12 /= 5; 
00461                   a13 /= 5;
00462                   a22 /= 5;
00463                   a23 /= 5; 
00464                   a33 /= 5;
00465 
00466                   matA1.at<float>(0, 0) = a11;
00467                   matA1.at<float>(0, 1) = a12;
00468                   matA1.at<float>(0, 2) = a13;
00469                   matA1.at<float>(1, 0) = a12;
00470                   matA1.at<float>(1, 1) = a22;
00471                   matA1.at<float>(1, 2) = a23;
00472                   matA1.at<float>(2, 0) = a13;
00473                   matA1.at<float>(2, 1) = a23;
00474                   matA1.at<float>(2, 2) = a33;
00475 
00476                   cv::eigen(matA1, matD1, matV1);
00477 
00478                   if (matD1.at<float>(0, 0) > 3 * matD1.at<float>(0, 1)) {
00479 
00480                     float x0 = pointSel.x;
00481                     float y0 = pointSel.y;
00482                     float z0 = pointSel.z;
00483                     float x1 = cx + 0.1 * matV1.at<float>(0, 0);
00484                     float y1 = cy + 0.1 * matV1.at<float>(0, 1);
00485                     float z1 = cz + 0.1 * matV1.at<float>(0, 2);
00486                     float x2 = cx - 0.1 * matV1.at<float>(0, 0);
00487                     float y2 = cy - 0.1 * matV1.at<float>(0, 1);
00488                     float z2 = cz - 0.1 * matV1.at<float>(0, 2);
00489 
00490                     float a012 = sqrt(((x0 - x1)*(y0 - y2) - (x0 - x2)*(y0 - y1))
00491                                * ((x0 - x1)*(y0 - y2) - (x0 - x2)*(y0 - y1)) 
00492                                + ((x0 - x1)*(z0 - z2) - (x0 - x2)*(z0 - z1))
00493                                * ((x0 - x1)*(z0 - z2) - (x0 - x2)*(z0 - z1)) 
00494                                + ((y0 - y1)*(z0 - z2) - (y0 - y2)*(z0 - z1))
00495                                * ((y0 - y1)*(z0 - z2) - (y0 - y2)*(z0 - z1)));
00496 
00497                     float l12 = sqrt((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2) + (z1 - z2)*(z1 - z2));
00498 
00499                     float la = ((y1 - y2)*((x0 - x1)*(y0 - y2) - (x0 - x2)*(y0 - y1)) 
00500                              + (z1 - z2)*((x0 - x1)*(z0 - z2) - (x0 - x2)*(z0 - z1))) / a012 / l12;
00501 
00502                     float lb = -((x1 - x2)*((x0 - x1)*(y0 - y2) - (x0 - x2)*(y0 - y1)) 
00503                              - (z1 - z2)*((y0 - y1)*(z0 - z2) - (y0 - y2)*(z0 - z1))) / a012 / l12;
00504 
00505                     float lc = -((x1 - x2)*((x0 - x1)*(z0 - z2) - (x0 - x2)*(z0 - z1)) 
00506                              + (y1 - y2)*((y0 - y1)*(z0 - z2) - (y0 - y2)*(z0 - z1))) / a012 / l12;
00507 
00508                     float ld2 = a012 / l12;
00509 
00510                     pointProj = pointSel;
00511                     pointProj.x -= la * ld2;
00512                     pointProj.y -= lb * ld2;
00513                     pointProj.z -= lc * ld2;
00514 
00515                     float s = 2 * (1 - 8 * fabs(ld2));
00516 
00517                     coeff.x = s * la;
00518                     coeff.y = s * lb;
00519                     coeff.z = s * lc;
00520                     coeff.h = s * ld2;
00521 
00522                     if (s > 0.4) {
00523                       laserCloudOri->push_back(pointOri);
00524                       //laserCloudSel->push_back(pointSel);
00525                       //laserCloudProj->push_back(pointProj);
00526                       //laserCloudCorr->push_back(laserCloudCornerFromMap->points[pointSearchInd[0]]);
00527                       //laserCloudCorr->push_back(laserCloudCornerFromMap->points[pointSearchInd[1]]);
00528                       //laserCloudCorr->push_back(laserCloudCornerFromMap->points[pointSearchInd[2]]);
00529                       //laserCloudCorr->push_back(laserCloudCornerFromMap->points[pointSearchInd[3]]);
00530                       //laserCloudCorr->push_back(laserCloudCornerFromMap->points[pointSearchInd[4]]);
00531                       coeffSel->push_back(coeff);
00532                     }
00533                   }
00534                 }
00535               }
00536             }
00537           }
00538           int laserCloudSelNum = laserCloudOri->points.size();
00539 
00540           float srx = sin(transformTobeMapped[0]);
00541           float crx = cos(transformTobeMapped[0]);
00542           float sry = sin(transformTobeMapped[1]);
00543           float cry = cos(transformTobeMapped[1]);
00544           float srz = sin(transformTobeMapped[2]);
00545           float crz = cos(transformTobeMapped[2]);
00546 
00547           if (laserCloudSelNum < 50) {
00548             continue;
00549           }
00550 
00551           cv::Mat matA(laserCloudSelNum, 6, CV_32F, cv::Scalar::all(0));
00552           cv::Mat matAt(6, laserCloudSelNum, CV_32F, cv::Scalar::all(0));
00553           cv::Mat matAtA(6, 6, CV_32F, cv::Scalar::all(0));
00554           cv::Mat matB(laserCloudSelNum, 1, CV_32F, cv::Scalar::all(0));
00555           cv::Mat matAtB(6, 1, CV_32F, cv::Scalar::all(0));
00556           cv::Mat matX(6, 1, CV_32F, cv::Scalar::all(0));
00557           for (int i = 0; i < laserCloudSelNum; i++) {
00558             pointOri = laserCloudOri->points[i];
00559             coeff = coeffSel->points[i];
00560 
00561             float arx = (crx*sry*srz*pointOri.x + crx*crz*sry*pointOri.y - srx*sry*pointOri.z) * coeff.x
00562                       + (-srx*srz*pointOri.x - crz*srx*pointOri.y - crx*pointOri.z) * coeff.y
00563                       + (crx*cry*srz*pointOri.x + crx*cry*crz*pointOri.y - cry*srx*pointOri.z) * coeff.z;
00564 
00565             float ary = ((cry*srx*srz - crz*sry)*pointOri.x 
00566                       + (sry*srz + cry*crz*srx)*pointOri.y + crx*cry*pointOri.z) * coeff.x
00567                       + ((-cry*crz - srx*sry*srz)*pointOri.x 
00568                       + (cry*srz - crz*srx*sry)*pointOri.y - crx*sry*pointOri.z) * coeff.z;
00569 
00570             float arz = ((crz*srx*sry - cry*srz)*pointOri.x + (-cry*crz - srx*sry*srz)*pointOri.y)*coeff.x
00571                       + (crx*crz*pointOri.x - crx*srz*pointOri.y) * coeff.y
00572                       + ((sry*srz + cry*crz*srx)*pointOri.x + (crz*sry - cry*srx*srz)*pointOri.y)*coeff.z;
00573 
00574             matA.at<float>(i, 0) = arx;
00575             matA.at<float>(i, 1) = ary;
00576             matA.at<float>(i, 2) = arz;
00577             matA.at<float>(i, 3) = coeff.x;
00578             matA.at<float>(i, 4) = coeff.y;
00579             matA.at<float>(i, 5) = coeff.z;
00580             matB.at<float>(i, 0) = -coeff.h;
00581           }
00582           cv::transpose(matA, matAt);
00583           matAtA = matAt * matA;
00584           matAtB = matAt * matB;
00585           cv::solve(matAtA, matAtB, matX, cv::DECOMP_QR);
00586 
00587           if (fabs(matX.at<float>(0, 0)) < 0.5 &&
00588               fabs(matX.at<float>(1, 0)) < 0.5 &&
00589               fabs(matX.at<float>(2, 0)) < 0.5 &&
00590               fabs(matX.at<float>(3, 0)) < 1 &&
00591               fabs(matX.at<float>(4, 0)) < 1 &&
00592               fabs(matX.at<float>(5, 0)) < 1) {
00593 
00594             transformTobeMapped[0] += matX.at<float>(0, 0);
00595             transformTobeMapped[1] += matX.at<float>(1, 0);
00596             transformTobeMapped[2] += matX.at<float>(2, 0);
00597             transformTobeMapped[3] += matX.at<float>(3, 0);
00598             transformTobeMapped[4] += matX.at<float>(4, 0);
00599             transformTobeMapped[5] += matX.at<float>(5, 0);
00600           } else {
00601             //ROS_INFO ("Mapping update out of bound");
00602           }
00603         }
00604       }
00605 
00606       transformUpdate();
00607 
00608       for (int i = 0; i < laserCloudLastNum; i++) {
00609         if (fabs(laserCloudLast->points[i].x) > 0.3 || fabs(laserCloudLast->points[i].y) > 0.3 ||
00610             fabs(laserCloudLast->points[i].z) > 0.3) {
00611 
00612           pointAssociateToMap(&laserCloudLast->points[i], &pointSel);
00613 
00614           int cubeI = int((pointSel.x + 10.0) / 20.0) + laserCloudCenWidth;
00615           int cubeJ = int((pointSel.y + 10.0) / 20.0) + laserCloudCenHeight;
00616           int cubeK = int((pointSel.z + 10.0) / 20.0) + laserCloudCenDepth;
00617           int cubeInd = cubeI + laserCloudWidth * cubeJ + laserCloudWidth * laserCloudHeight * cubeK;
00618 
00619           laserCloudArray[cubeInd]->push_back(pointSel);
00620         }
00621       }
00622 
00623       for (int i = 0; i < laserCloudValidNum; i++) {
00624         laserCloudCorner->clear();
00625         laserCloudSurf->clear();
00626         pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudCubePointer = 
00627                                                laserCloudArray[laserCloudValidInd[i]];
00628         int laserCloudCubeNum = laserCloudCubePointer->points.size();
00629         for (int j = 0; j < laserCloudCubeNum; j++) {
00630           if (fabs(laserCloudCubePointer->points[j].v - 2) < 0.005 || 
00631               fabs(laserCloudCubePointer->points[j].v - 1) < 0.005) {
00632             laserCloudCorner->push_back(laserCloudCubePointer->points[j]);
00633           } else {
00634             laserCloudSurf->push_back(laserCloudCubePointer->points[j]);
00635           }
00636         }
00637 
00638         laserCloudCorner2->clear();
00639         pcl::VoxelGrid<pcl::PointXYZHSV> downSizeFilter;
00640         downSizeFilter.setInputCloud(laserCloudCorner);
00641         downSizeFilter.setLeafSize(0.05, 0.05, 0.05);
00642         downSizeFilter.filter(*laserCloudCorner2);
00643 
00644         laserCloudSurf2->clear();
00645         downSizeFilter.setInputCloud(laserCloudSurf);
00646         downSizeFilter.setLeafSize(0.1, 0.1, 0.1);
00647         downSizeFilter.filter(*laserCloudSurf2);
00648 
00649         laserCloudCubePointer->clear();
00650         *laserCloudCubePointer = *laserCloudCorner2 + *laserCloudSurf2;
00651       }
00652 
00653       laserCloudSurround->clear();
00654       for (int i = 0; i < laserCloudSurroundNum; i++) {
00655          pcl::PointCloud<pcl::PointXYZHSV>::Ptr laserCloudCubePointer = 
00656                                                 laserCloudArray[laserCloudSurroundInd[i]];
00657          int laserCloudCubeNum = laserCloudCubePointer->points.size();
00658          for (int j = 0; j < laserCloudCubeNum; j++) {
00659            pointSurround.x = laserCloudCubePointer->points[j].x;
00660            pointSurround.y = laserCloudCubePointer->points[j].y;
00661            pointSurround.z = laserCloudCubePointer->points[j].z;
00662            pointSurround.intensity = laserCloudCubePointer->points[j].h;
00663            laserCloudSurround->push_back(pointSurround);
00664          }
00665       }
00666 
00667       sensor_msgs::PointCloud2 laserCloudSurround2;
00668       pcl::toROSMsg(*laserCloudSurround, laserCloudSurround2);
00669       laserCloudSurround2.header.stamp = ros::Time().fromSec(timeLaserCloudLast);
00670       laserCloudSurround2.header.frame_id = "/camera_init_2";
00671       pubLaserCloudSurround.publish(laserCloudSurround2);
00672 
00673       geometry_msgs::Quaternion geoQuat = tf::createQuaternionMsgFromRollPitchYaw
00674                      (transformBefMapped[2], -transformBefMapped[0], -transformBefMapped[1]);
00675 
00676       odomBefMapped.header.stamp = ros::Time().fromSec(timeLaserCloudLast);
00677       odomBefMapped.pose.pose.orientation.x = -geoQuat.y;
00678       odomBefMapped.pose.pose.orientation.y = -geoQuat.z;
00679       odomBefMapped.pose.pose.orientation.z = geoQuat.x;
00680       odomBefMapped.pose.pose.orientation.w = geoQuat.w;
00681       odomBefMapped.pose.pose.position.x = transformBefMapped[3];
00682       odomBefMapped.pose.pose.position.y = transformBefMapped[4];
00683       odomBefMapped.pose.pose.position.z = transformBefMapped[5];
00684       pubOdomBefMapped.publish(odomBefMapped);
00685 
00686       geoQuat = tf::createQuaternionMsgFromRollPitchYaw
00687                 (transformAftMapped[2], -transformAftMapped[0], -transformAftMapped[1]);
00688 
00689       odomAftMapped.header.stamp = ros::Time().fromSec(timeLaserCloudLast);
00690       odomAftMapped.pose.pose.orientation.x = -geoQuat.y;
00691       odomAftMapped.pose.pose.orientation.y = -geoQuat.z;
00692       odomAftMapped.pose.pose.orientation.z = geoQuat.x;
00693       odomAftMapped.pose.pose.orientation.w = geoQuat.w;
00694       odomAftMapped.pose.pose.position.x = transformAftMapped[3];
00695       odomAftMapped.pose.pose.position.y = transformAftMapped[4];
00696       odomAftMapped.pose.pose.position.z = transformAftMapped[5];
00697       pubOdomAftMapped.publish(odomAftMapped);
00698 
00699       /*sensor_msgs::PointCloud2 pc12;
00700       pcl::toROSMsg(*laserCloudCornerFromMap, pc12);
00701       pc12.header.stamp = ros::Time().fromSec(timeLaserCloudLast);
00702       pc12.header.frame_id = "/camera_init_2";
00703       pub1.publish(pc12);
00704 
00705       sensor_msgs::PointCloud2 pc22;
00706       pcl::toROSMsg(*laserCloudSel, pc22);
00707       pc22.header.stamp = ros::Time().fromSec(timeLaserCloudLast);
00708       pc22.header.frame_id = "/camera_init_2";
00709       pub2.publish(pc22);
00710 
00711       sensor_msgs::PointCloud2 pc32;
00712       pcl::toROSMsg(*laserCloudCorr, pc32);
00713       pc32.header.stamp = ros::Time().fromSec(timeLaserCloudLast);
00714       pc32.header.frame_id = "/camera_init_2";
00715       pub3.publish(pc32);
00716 
00717       sensor_msgs::PointCloud2 pc42;
00718       pcl::toROSMsg(*laserCloudProj, pc42);
00719       pc42.header.stamp = ros::Time().fromSec(timeLaserCloudLast);
00720       pc42.header.frame_id = "/camera_init_2";
00721       pub4.publish(pc42);*/
00722     }
00723 
00724     status = ros::ok();
00725     cvWaitKey(10);
00726   }
00727 
00728   return 0;
00729 }


loam_back_and_forth
Author(s): Ji Zhang
autogenerated on Tue Jan 7 2014 11:20:25