Self-Driving Car Engineer Nanodegree Program
In this project, the goal is to safely navigate around a virtual highway with other traffic that is driving +-10 MPH
of the 50 MPH speed limit
. The simulator provides the vehicle's localization and sensor fusion data; there is also a
sparse map list of waypoints around the highway. The vehicle (from now on referred to as the Ego) should try to go
as close as possible to the 50 MPH speed limit
, which means passing slower traffic when possible, while other cars
(from now on referred to as the Bees) may change lanes too.
The Ego should avoid colliding with the bees at all cost as well as driving inside of the marked road lanes at all times unless going from one lane to another. The Ego should be able to make one complete loop around the 6946 m
highway loop. Since it is trying to go 50 MPH
, it should take a little over 5
minutes to complete one loop. Also, the Ego should not experience total acceleration over 10
m/s^2 and jerk that is greater than 10
m/s^3.
The map data of the highway is stored in data/highway_map.txt
.
Each waypoint in the list contains [x,y,s,dx,dy]
values. x
and y
are the waypoint's Cartesian coordinates on
the map; the s
value is the distance along the road to get to that waypoint in meters, the dx
and dy
values define the unit normal vector pointing outward of the highway loop.
The highway's waypoints loop around so the frenet s value, a distance along the road, goes from 0
to 6945.554
.
The power of Spline functionality has been heavily leveraged throughout the project.
Among the 5 elements of the waypoint data, s
is guaranteed not to have duplicate values. Thus, 4 tk::spline
objects
have been created for x
, y
, dx
and dy
with respect to s
. Those splines are used by the instance of
the PathAgent
class to obtain the Cartesian ({x, y}
) coordinates from the Frenet ({s, d}
).
- Obtain current Ego telemetry, sensor fusion data, and information of the remainder of the previous path.
- Preserve the state of the Ego at the end of the previous path. It is being used as a starting point for computing the path extension.
- Store the forward proximity for each of three lanes in a
vector
of structuredProxy
states. - Evaluate the attractiveness of each lane depending on its free forward space and remoteness from the current lane.
- Recursively evaluate lanes for safety (absence of collisions), starting from the most attractive to the least attractive
until the safe lane is found. If no reliable lane determined other than the current lane, the latter is
picked and its safety being enforced by following the
proxy
at a safe distance. - Generate new path, attach it to the previously unused path and send it to the Simulator.
StateAgent
class is responsible for points 1 - 3, while PathAgent
performs 4 - 6.
In my opinion, the task at hand is an excellent fit for Q-Learning, with the perfect coverage of our domain-specific implementation by Lex Fridman. But, apart from the colossal pain it takes to make TensorFlow work in C++ (while it is itself written in C++), the Simulator interface makes the task almost prohibitively hard.
I have thoroughly studied the Final State Machine approach covered in classes. To say the least, this is not my style of decision-making.
The algorithm that I've finally derived continually scans through the available lanes and picks the most attractive among those that are collision-free.
For trajectory, I have extensively borrowed the techniques used by Aaron Brown in his native implementation. In fact, I do have a successful implementation using the concept of Jerk Minimizing Trajectories provided in classes but failed to prevent it from breaking the jerk/acceleration limits in some marginal states. Aaron is presumably the co-author of the Simulator and its interface with the outer world, so he probably knows the best way to interact with it.
The implementation more or less guarantees that no incidents will be triggered by the Ego. It also processes the situations when Bee unexpectedly swerves into the Ego's lane right in front of it.
However, situations like the one below aren't processed.
This looks like a common self-driving car problem when human drivers rear-end autonomous vehicles and is beyond the scope of this Project specification, although it obviously possible to properly address such situations.
The sample video of successfully passing the entire highway loop may be found on my humble YouTube channel.
I've added two new sources to CMakeLists.txt
: src/stateAgent.cpp
and src/pathAgent.cpp
mkdir build && cd build
cmake .. && make
./path_planning
.