Team Members | Role | Slack | github | |
---|---|---|---|---|
Alan Hekle | [email protected] | Team Leader / Corrections / PEP8 / Write-up | @theaisback | TheAisBack |
Samir Haddad | [email protected] | TL Detector / Twist Controller | @sam01 | SamH1 |
Kev Lai | [email protected] | Waypoint / Twist Controller | @kevlai | kevguy |
Brian McHugh | [email protected] | Continous Integration / Floater | @mchugh | Brian-Leary |
Lucas Meyer | [email protected] | TL Detector / Twist Controller/ Waypoint | @lcmeyer | lcmeyer37 |
This is the project repo for the final project of the Udacity Self-Driving Car Nanodegree: Programming a Real Self-Driving Car. For more information about the project, see the project introduction here.
The objective of this project is to use different concepts learned and applied during the course of the Self-Driving Car Engineer Nanodegree Program to implement a car that is able to drive autonomously. Traffic lights, translation and rotation variables are taken into consideration, and also the comfort of the passengers inside the car, such as maximum jerk, acceleration and velocity values were applied while implementing the car controls.
The project uses the GitHub repository provided by Udacity here, a VM provided by the instructors and a Unity3D simulator that presents the highway, traffic lights, and the car as testing environment for the code.
This project makes use of the Robot Operating System (ROS), a middleware collection of frameworks for robot software development. For this project a system architecture was followed, interpreting the car controller as subsystems: Perception, Planning and Control, and each subsystem is created making use of Python scripts, ROS nodes and topics, and other provided code from the repository.
Here is the architecture diagram of the self-driving car, which will help us create the software to run the actual car.
The System Architecture comprises of the following subsystems:
Subsystem responsible for the detection of the traffic lights, the car should be able to visualize the traffic light signs and characterize it as a green, red or yellow light. The folder tl_detector in this repo contains the files needed for implementation of this subsystem:
This python script is basically responsible for calling the TLClassifier class, that is able to understand the traffic light signs, and contains the following ROS subscribers and publishers:
Subscription, current position of the car.
Subscription, waypoints of the simulator track.
Subscription, location of the traffic lights in 3D map space.
Subscription, contains the image captured by the car's camera for traffic light classification.
Publisher, contains the status of the traffic lights.
Python script that implements the TLClassifier class, the following methods for this class are worth mentioning:
This function is responsible for loading a Deep Neural Network that is being used for the traffic lights classification, the SSD w/ MobileNet trained with the COCO dataset. This neural network is presented by the Tensorflow Object Detection API, and contains dozens of different object classes for detection. For this project, just class '10' is of interest, the traffic light object class.
Responsible for using the image provided by the car's camera to locate the traffic lights on it, making use of the Neural Network described above. This function returns rectangles, which surround and gives information of the position of the traffic light detected.
Using the rectangle provided above, the tl_detector.py script separates and resizes a new image 32x32 that contains just the traffic light sign. That image is used in this function, which basically applies three HSV masks for green, yellow and red. Then, these masks have their pixels counted, and the mask that has more pixels probably represents the color of the traffic light. For example, if the mask_red has more pixels than mask_yellow or mask_green, probably the light is red.
The subsystem is responsible for taking into consideration the information provided by the Perception subsystem, and current car information, to decide a further course of action for the car. The folder waypoint_updater contains one python script used for that matter.
This python script uses the following:
Subscription.
Subscription.
Subscription.
Subscription, contains the current velocity of the car.
Publisher, contains the list of waypoints that are given to the Control subsystem, that uses it to drive the car.
This function is worth mentioning, it basically uses the subscriptions described above to return the list of waypoints to be published to '/final_waypoints'. This function starts by finding the closest waypoint to the vehicle, next this waypoint is considered first in the following list of next waypoints, then the traffic light location and its sign status is used to further decide if the car should be stopping, or accelerating, or maintaining velocity. Finally, these decisions are all implemented as the next waypoints list that is returned by this function.
Subsystem responsible for using the waypoints provided by the Planning subsystem, and current car information, to publish messages that are used by the simulator, related directly to control of the car. The folder twist_controller contains python scripts related to this subsystem:
This python script uses several ros parameters, subscribers and publishers such as:
Contains the car mass.
Fuel capacity of the car.
Maximum lateral acceleration of the car.
Maximum steering angle of the car.
Kp value for PID controller of the throttle.
Ki value for PID controller of the throttle.
Kd value for PID controller of the throttle.
Subscription.
Subscription, contains information if the Drive-by-wire (DBW) system is enabled.
Subscription, published by the waypoint_follower/pure_pursuit.cpp provided script.
Publisher, contains a SteeringCmd type ROS message, that is used by the simulator for steering of the car.
Publisher, contains a ThrottleCmd type ROS message, that is used by the simulator for throttling of the car.
Publisher, contains a BrakeCmd type ROS message, that is used by the simulator for braking of the car.
Worth mentioning, this function uses the throttle, brake and steering variables provided by the Controller class from twist_controller.py to preprocess and decide which control variables are of interest and then published. For example, normally a driver wouldn't use the brake and throttle pedals at the same time. So, in this function, it first sees if the throttle provided by the controller in non-zero, if so, just throttle and steering are published. Otherwise, just the brake command will be published along with the steering.
This python script uses the PID controller provided by Udacity, the Proportional-Integral-Derivative controller is a control loop feedback mechanism used in industrial control systems that require continuously modulated control. The PID controller calculates an error value as the difference between a desired setpoint and a measured process variable, and applies a correction based on proportional, integral and derivative terms. This controller is one of the topics explained on this course on Term 2: Sensor Fusion, Localization and Control.
This function is worth mentioning because it calculates the errors described above and uses the PID class provided to calculate the next value for throttle and brake. The steering value uses the YawController class to get the steering angle needed for driving.
Here is our output from the simulator.
Considering that the car currently is able to drive itself along the simulator, with an average speed of approximately 10 miles per hour, and that the car is able to follow the path of the highway and detect traffic lights, deciding if it should stop or go along the course, the group has decided that the current state of the project is acceptable as resolution and for presentation.
-
Be sure that your workstation is running Ubuntu 16.04 Xenial Xerus or Ubuntu 14.04 Trusty Tahir. Ubuntu downloads can be found here.
-
If using a Virtual Machine to install Ubuntu, use the following configuration as minimum: Virtual Box
- 2 CPU
- 2 GB system memory
- 25 GB of free hard drive space
The Udacity provided virtual machine has ROS and Dataspeed DBW already installed, so you can skip the next two steps if you are using this.
-
Follow these instructions to install ROS
- ROS Kinetic if you have Ubuntu 16.04.
- ROS Indigo if you have Ubuntu 14.04.
-
- Use this option to install the SDK on a workstation that already has ROS installed: One Line SDK Install (binary)
-
Download the Udacity Simulator.
Build the docker container
docker build . -t capstone
If the above terminal code does not work type...
sudo docker build . -t capstone
Run the docker file
docker run -p 4567:4567 -v $PWD:/capstone -v /tmp/log:/root/.ros/ --rm -it capstone
- Clone the project repository
git clone https://github.com/udacity/CarND-Capstone.git
- Install python dependencies
cd CarND-Capstone
pip install -r requirements.txt
- Make and run styx
cd ros
catkin_make
source devel/setup.sh
roslaunch launch/styx.launch
- Run the simulator
- Download training bag that was recorded on the Udacity self-driving car (a bag demonstraing the correct predictions in autonomous mode can be found here)
- Unzip the file
unzip traffic_light_bag_files.zip
- Play the bag file
rosbag play -l traffic_light_bag_files/loop_with_traffic_light.bag
- Launch your project in site mode
cd CarND-Capstone/ros
roslaunch launch/site.launch
- Confirm that traffic light detection works on real life images