/sfnd-3d-object-tracking

Sensor Fusion, Project 3: 3D Object Tracking

Primary LanguageC++OtherNOASSERTION

SFND 3D Object Tracking

Final TTC estimation

This is the final project for the camera unit within Udacity's Sensor Fusion Nanodegree. This project covers the following key concepts:

  • Keypoint detectors and descriptors
  • Object detection using the pre-trained YOLO deep-learning framework
  • Methods to track objects by matching keypoints and bounding boxes across successive images
  • Associating regions in a camera image with lidar points in 3D space

The flowchart below provides an overview of the final project structure. The steps in the orange box were completed in the previous project 2D Feature Tracking. This project builds on that previous one, implementing the steps in the blue box and downstream. Objects detected with the YOLO deep neural network are tracked across frames by considering the strength of keypoint correspondences within their bounding boxes. Finally, a robust estimation of time-to-collision (TTC) is performed with data from both the lidar and camera sensors.

Dependencies

Building and running the project

mkdir build && cd build
cmake ..
make
./3D_object_tracking

Writeup, Task FP.0

FP.1 Match 3D objects

Lines 224-284 in camFusion_Student.cpp
As suggested, I used a std::multimap<int,int> to track pairs of bounding box IDs. I then counted the keypoint correspondences per box pair to determine the best matches between frames.

// Count the greatest number of matches in the multimap, where each element is {key=currBoxID, val=prevBoxID}

FP.2 Compute lidar-based TTC

Lines 202-221 in camFusion_Student.cpp
In each frame, I took the median x-distance to reduce the impact of outlier lidar points on my TTC estimate. With the constant velocity model, the key equation is as follows.

TTC = d1 * (1.0 / frameRate) / (d0 - d1);

Lines 192-199 in camFusion_Student.cpp
To calculate the median, I built a helper function to sort the vector of lidar points.

void sortLidarPointsX(std::vector<LidarPoint> &lidarPoints)
{
    // This std::sort with a lambda mutates lidarPoints, a vector of LidarPoint
    std::sort(lidarPoints.begin(), lidarPoints.end(), [](LidarPoint a, LidarPoint b) {
        return a.x < b.x;  // Sort ascending on the x coordinate only
    });
}

FP.3 Associate keypoint matches with bounding boxes

Lines 133-142 in camFusion_Student.cpp
This function is called for each bounding box, and it loops through every matched keypoint pair in an image. If the keypoint falls within the bounding box region-of-interest (ROI) in the current frame, the keypoint match is associated with the current BoundingBox data structure.

FP.4 Compute mono camera-based TTC

Lines 145-189 in camFusion_Student.cpp
The code for this function computeTTCCamera draws heavily on the example provided in an earlier lesson. It uses distance ratios on keypoints matched between frames to determine the rate of scale change within an image. This rate of scale change can be used to estimate the TTC.

TTC = (-1.0 / frameRate) / (1 - medianDistRatio);

Like the lidar TTC estimation, this function uses the median distance ratio to avoid the impact of outliers. Unfortunately this approach is still vulnerable to wild miscalculations (-inf, NaN, etc.) if there are too many mismatched keypoints. Note also that this algorithm calculates the Euclidean distance for every paired combination of keypoints within the bounding box, O(n^2) on the number of keypoints.

FP.5 Performance evaluation, lidar outliers

I was not able to find any frames where the lidar estimated TTC was unreasonable. It ranged from about 8-15 seconds. I believe the approach of taking the median point, rather than the closest point, has avoided the problem introduced by outliers. I've included some examples of the lidar top-view below.

Figure 1, near-side outlier

Figure 2, far-side outlier

Figure 3, multiple outliers

Figure 4, a tight group without outliers

FP.6 Performance evaluation, detector/descriptor combinations

Certain detector/descriptor combinations, especially the Harris and ORB detectors, produced very unreliable camera TTC estimates. Others, such as SIFT, FAST, and AKAZE detectors produced reliable results in line with the TTC estimates provided by the lidar sensor alone. Below is a sample of the results matrix, ranked by TTC estimate difference (between the camera and lidar systems).

Selection from results matrix

As an example, here is some output from the AKAZE detector/descriptor combination with the boolean perf flag set:

$ ./3D_object_tracking

Detector,Descriptor,Frame index,TTC lidar,TTC camera,TTC difference

AKAZE,AKAZE,1,12.5156,12.3444,-0.171208,
AKAZE,AKAZE,2,12.6142,14.0983,1.48408,
AKAZE,AKAZE,3,14.091,12.8828,-1.20818,
AKAZE,AKAZE,4,16.6894,14.4781,-2.21132,
AKAZE,AKAZE,5,15.9082,16.7468,0.838524,
AKAZE,AKAZE,6,12.6787,13.8675,1.18879,
AKAZE,AKAZE,7,11.9844,15.3448,3.36044,
AKAZE,AKAZE,8,13.1241,14.1378,1.01365,
AKAZE,AKAZE,9,13.0241,13.8398,0.815652,
AKAZE,AKAZE,10,11.1746,11.5483,0.373644,
AKAZE,AKAZE,11,12.8086,12.1341,-0.674506,
AKAZE,AKAZE,12,8.95978,11.0554,2.09557,
AKAZE,AKAZE,13,9.96439,11.2852,1.32083,
AKAZE,AKAZE,14,9.59863,10.5842,0.985564,
AKAZE,AKAZE,15,8.57352,10.1989,1.62537,
AKAZE,AKAZE,16,9.51617,9.81256,0.296389,
AKAZE,AKAZE,17,9.54658,9.06452,-0.482057,
AKAZE,AKAZE,18,8.3988,8.97386,0.575053,