Skip to content

Udacity Self-Driving Car Engineer Nanodegree - Term1 Project 3 (Behavior Cloning)

Notifications You must be signed in to change notification settings

jamesjosephreynolds/BehaviorCloning

Repository files navigation

Project 3 - Driver Behavior Cloning

Introduction

For this project I decided to maintain two Python scripts: model.py with all of the CNN model functionality, and custom_functions.py, which contains many functions utilized in the model.py. This makes model.py much simpler to read and navigate. Please be sure to include custom_functions.py before attempting to run model.py.

I used the data provided by Udacity to train my model. I initially tried to generate my own data, with both the stable and beta versions of the simulator, but I could not control the car smoothly in either case, without the benefit of a steering wheel or joystick.

Pre-Processing

For my image pre-processing, I leaned heavily on the separate blogposts by Vivek Yadav and Mojtaba Valipour (my Udacity mentor). Without these resources I fear I would have floundered for many days before I was able to train my model for any semblance of control.

My utility function, pre_process(), uses two possible operating modes: ‘train’ or None. This allows me to use the same pre-processing function call for both the training and validation generator, as well as in the driving script, drive.py. For the latter, I set the mode = None. In this case, the only function is to crop the hood and part of the sky out of the image, then resize the image to 66x200 pixels, around the center pixel, and normalize the data. For training, and for validation, there are several other steps, as follows:

  1. The image is randomly translated vertically
  2. The image is randomly translated horizontally
  3. The image brightness is randomly skewed (in HSV colorspace)
  4. The image is randomly flipped horizontally (50% probability). In the case of horizontal translation, a gain is applied to the magnitude of the translation, and this added to the input steering angle. As the image is translated further and further, the steering angle adjustment is greater. In the case of horizontal flipping, the steering angle is also flipped (multiple by -1).

The main way that my implementation is unique from those blogposts above, is that following the translations, during the cropping step, I remove the black space from the final image, by not including it in the cropped area. As far as I see, both of the blogposts above leave this black space in the image as, essentially, noise. I felt this was unnecessary, as the final image size is much smaller than the original, leaving some room to remove the black space. A sample of my augmentation for a single image is shown here:

augmentation.png should go here, whoops!

Multiple results of randomized data augmentation for a single image

Early on my models were not able to navigate sharp corners with non-grassy backgrounds (water or dirt). I realized that I had cropped off too much of the top of the image, and the model was having trouble distinguishing those areas from the road. I updated my pre-processing to leave more of the horizon visible. This final level of cropping is represented above.

CNN Model Architecture

My model architecture is based on that of Nvidia in their end-to-end paper. The differences are as follows: I did the normalization outside of the CNN with an explicit function, and rather than using multi-pixel strides during convolution, I used max pooling layers after the convolution. It’s not clear if the authors used dropout, but as many references (blogposts above, Confluence posts, etc) cautioned about overfitting in this project, I included dropout after every layer.

For the last layer I chose ‘tanh’ for my activation layer, to ensure I could get results between -1 and +1. For all other layers I chose ‘relu’.

Layer (type) Output Shape Param # Connected to
convolution2d_1 (Convolution2D) (None, 62, 196, 24) 1824 convolution2d_input_1[0][0]
maxpooling2d_1 (MaxPooling2D) (None, 31, 98, 24) 0 convolution2d_1[0][0]
dropout_1 (Dropout) (None, 31, 98, 24) 0 maxpooling2d_1[0][0]
convolution2d_2 (Convolution2D) (None, 27, 94, 36) 21636 dropout_1[0][0]
maxpooling2d_2 (MaxPooling2D) (None, 14, 47, 36) 0 convolution2d_2[0][0]
dropout_2 (Dropout) (None, 14, 47, 36) 0 maxpooling2d_2[0][0]
convolution2d_3 (Convolution2D) (None, 10, 43, 48) 43248 dropout_2[0][0]
maxpooling2d_3 (MaxPooling2D) (None, 5, 22, 48) 0 convolution2d_3[0][0]
dropout_3 (Dropout) (None, 5, 22, 48) 0 maxpooling2d_3[0][0]
convolution2d_4 (Convolution2D) (None, 3, 20, 64) 27712 dropout_3[0][0]
dropout_4 (Dropout) (None, 3, 20, 64) 0 convolution2d_4[0][0]
convolution2d_5 (Convolution2D) (None, 1, 18, 64) 36928 dropout_4[0][0]
dropout_5 (Dropout) (None, 1, 18, 64) 0 convolution2d_5[0][0]
flatten_1 (Flatten) (None, 1152) 0 dropout_5[0][0]
dense_1 (Dense) (None, 100) 115300 flatten_1[0][0]
dropout_6 (Dropout) (None, 100) 0 dense_1[0][0]
dense_2 (Dense) (None, 50) 5050 dropout_6[0][0]
dropout_7 (Dropout) (None, 50) 0 dense_2[0][0]
dense_3 (Dense) (None, 10) 510 dropout_7[0][0]
dropout_8 (Dropout) (None, 10) 0 dense_3[0][0]
dense_4 (Dense) (None, 1) 11 dropout_8[0][0]

Trainable params: 252,219

CNN_architecture.png should go here, whoops!

Graphical representation of the architecture

Training

For training, I divided the Udacity dataset into both training and validation sets, with an 80/20 split. I didn’t explicitly make a test dataset, as the real test is whether the car is able to drive around the track in the simulator. Testing the model on an image dataset in the Keras environment seems a pointless exercise.

My original attempts at training the model were unsuccessful, as the car seemed to drive almost perfectly straight all the time. I suspect this is due the majority of the data having zero steering angle, and the model tending to overfit to this situation. This is one difficulty of a mean-square error optimization when most of the output values are the same. My first mitigation strategy for this was to shift most images with zero steering angle to the left or right, and give a nonzero command. This is implemented via the custom functions horizontal_shift() and pre_process(). After training this model a couple of times with decreasing learning rate, I was able to get a model that cleared the first five or so turns on the track. At the second sharp turn before a dirt patch, the car drove straight off the road again. My next mitigation technique was to train again, but to drop 90% of the images where the steering angle is less than 0.1. This approach allowed me to get a trained model that would navigate the entire track.

My final solution was able to navigate track 1 without leaving the safe surface. The steering is bit oscillatory, owing to my having dropped most of the images with zero steering angle. I did a fine tuning whereby I allowed more of the zero angle images into the training set, and the issue was reduced, but still visible. I tried this same refinement again, but eventually, although the oscillations were damped out, the car tended to drive straight into the first dirt area again. So, I settled for the oscillating solution, not because it's the best solution, but because this project was quite time consuming, and I decided it was good enough for now. My main takeaways from this tuning and retuning were that there is a very delicate balance to this approach, when no sensor input outside of a camera is used, and that one should take great care to same model weight files into a version control system, so as not to lose a well-behaved model and not have the ability to retrieve it.

Reflections

While it seems possible to train a car to drive in this way, it obviously has some very severe limitations that need to be considered. For instance, obstacle avoidance is probably much better handled by an active object detection system, and emergency control, than by vision-based driving. Another example is understanding road conditions: wet, icy, sandy, etc. Obviously the vehicle handling is dependent on the surface over which it is driving, so understanding this variable is critical. In other words, even if this method can account for 99.9% of situations, that is not nearly good enough.

Another interesting behavior that I noted, in an early build of my model, was that the car would swerve around the shadow cast by a tree after the first turn. Eventually this was removed through further training, but it is clear that the difference in brightness caused the car to think that this was not an acceptable part of the road on which to drive. This is an area that I hope we will explore further in the Advanced Lane Finding project.

Lastly, I would like to say that this project was very challenging, and I found the native Udacity materials inadequate. However, since Udacity linked Paul Heraty's Confluence post, I was able to plow through it. I'm thankful this information was available and suggest it be included in the course material a priori for future cohorts.

About

Udacity Self-Driving Car Engineer Nanodegree - Term1 Project 3 (Behavior Cloning)

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages