The project focuses on collecting and analyzing hardware benchmarks while training a deep learning network. Training deep learning networks is a resource-intensive and time-consuming process. Building a well-performing pipeline requires identifying low throughput areas and mitigating them with either software tuning or hardware upgrades. Low throughput areas depend on the variability of chosen models, data types, and individual programming styles. Objective benchmark measurement is necessary in order to successfully address heavy system load and uneven utilization of the underlying components. The project consists of a Bash shell script, which collects five system metrics: processor temperature, processor utilization, disk input/output utilization, memory usage, and GPU temperature. Each metric is collected while training Tensorflow's object detection (faster R-CNN with inception) model to look for Waldo. The metrics are visualized via an interactive R Shiny application.
System Administration and Engineering is a profession tasked with continuous evaluation and improvement of system performance. Kernel and end-user processes generate utilization spikes. Frequently, the spikes are uneven with a particular system resource causing a system-wide slowdown. Shared servers used for training deep learning networks are in need of constant optimization, which is frequently unique to a specific training type. Having a shell script, which can collect system utilization paired with an interactive R Shiny application will help System Engineers perform custom infrastructure tuning.
Currently, I am a Senior Systems Administrator at a research university. My infrastructure has been expanding. I am involved in helping laboratories collect video data and use Tensorflow (DeepLabCut) for video analysis. I am frequently asked if investing in a CUDA-compatible graphics card will accelerate network training. The obvious answer is, "yes, it will". However, I always wanted to make this answer clear and quantifiable. Dedicated GPUs are expensive. They consume more electricity and generate additional heat. However, when dealing with Tensorflow and object detection models, how exactly do GPUs accelerate machine learning pipelines and affect system hardware utilization?
Null Hypothesis
Part A: Training Tensorflow deep neural network on the CPU vs. the GPU does not yield any quantifiably observable diffirences. Hardware utilization is objectively the same.
Part B: Training Tensorflow object detection model on the CPU vs. the GPU does not accelerate research process.
Alternative Hypothesis
Part A: Training Tensorflow deep neural network on the CPU vs. the GPU yields different hardware utilization patterns, which are clearly quantifiable and observable.
Part B: Training Tensorflow object detection model on the CPU vs. GPU has potential to accelerate research process.
Tensorflow has several own benchmark utilities. I have unsuccessfully tried working with them. I was surprised to find very little documentation on how to use these benchmarking instruments. Additionally, I was bewildered by the fact that tf_cnn_benchmark required installation and configuration of Google cloud services. I could not find a good reason to install and configure Google cloud services on my research servers just to benchmark Tensorflow. Finally, it did not seem that anything was collecting disk, memory, processor, and temperature at the same (and equal) time intervals. Hence, I have finally decided to write my own script.
I work at a research facility, which uses Tensorflow for markerless tracking of primates. Rules of engagement dictate that I cannot share or distribute the media collected at the facility. Hence, I had to find a proxy object detection mechanism. Being a dad of two boys, I was elated to stumble upon Tadej Magajna's HereIsWally project. There is no harm in working while having fun looking for Waldo, is there?
The script I wrote uses Bourne Again Shell (BASH) to set several variables, train Tensorflow network for a set amount of time, collect hardware metrics, and export the resulting Tensorflow chekpoints into pre-trained inference graphs. In theory, it should not be difficult to modify the script to execute a different model or run on different hardware. The repository has two versions of the script, since my hypotheses deal with GPU vs. CPU training comparisons. I wanted to have a clear way to run the script (and save the results into different directories). collect_cpu.sh trains the model on the processor while collect_gpu.sh does the same using the GPU.
The GPU card, which was used for this project is GeForce RTX 2070 with Max-Q Design. CUDA libraries were failing to load from inside of Tensorflow until I added the following line:
config.gpu_options.allow_growth = True
Current version Tensorflow suppresses output by default, so I also had to add the following two lines to be able to see loss values:
tf.logging.set_verbosity(tf.logging.INFO)
config = tf.ConfigProto()
sess = tf.Session(config=config)
I also wanted to save checkpoints every 60 seconds because my shell script aborted the training after 75 minutes. Hence, I added the save_checkpoints_secs=60 parameter:
config = tf.estimator.RunConfig(model_dir=FLAGS.model_dir, save_checkpoints_secs=60)
These scripts are modified versions of Tadej's find_wally_pretty.sh. I wanted to be able to find Waldo in a scripted manner saving output files into a separate directory. Threfore, I've added the following to the bottom of the file:
plt.axis("off")
fig = ax.imshow(image_np)
orig = args.image_path
p = re.compile(r'(\d+\.\w{3}$)')
m = p.search(orig)
filename = file_output + m.group(1)
fig.axes.get_xaxis().set_visible(False)
fig.axes.get_yaxis().set_visible(False)
plt.savefig(filename,bbox_inches="tight",pad_inches = 0)
plt.close()
I also had to import regular expression library and set inference graph location and output directories:
import re
model_path = '/home/motorns/Documents/datascience/finalproject/waldo/inferenceGraphCPU/frozen_inference_graph.pb'
file_output = './images/cpu/'
I modified confidence score to be low, so I could see Tensorflow making mistakes:
if scores[0][0] < 0.0001:
sys.exit('Wally not found :(')
Finally, I was able to run this simple For Loop:
for i in *jpg; do
python3 find_wally_GPU.sh $i
done
The loop goes through Wally images looking for him and saving the resulting matplotlib figures into files with the same names but corresponding (cpu or gpu) directories.
This script was tested on the following hardware
Gigabyte Aero 15x Laptop | |||
Processor | RAM | Graphics Card | Hard Drive |
Intel 6-core i9-8950HK CPU @ 2.90GHz | 32 GB, Speed: 2667 MT/s | GeForce RTX 2070 with Max-Q Design | Intel SSD Pro 7600p/760p/E 6100p Series |
Anaconda environment file contains remnants of trying (and failing) to use Tensorflow's own benchmarking utilities. However, it does work and is needed to train the network. It can be created:
conda env create -f benchmark.yml
The command above will create the environment named bench, which can then be imported:
conda activate bench
Every control of the R Shiny application works with every tabbed ggplot graph. Selecting several hardware components at once adds a side-by-side comparison panel. Feel free to inspect interactivity screenshots for examples.
CPU-trained model was a lot less accurate as seen in the app because 75 minutes of training translated into only 3,421 steps. When training for the same time on the GPU, the model was able to complete 34,264 steps (ten times as many). As also seen in the app, processor utilization is minimal when GPU-based training is taking place. Hece, in theory, GPU-based training is beneficial on a shared server. While the graphics card is doing the heavy lifting, the processor is free to perform other tasks. One surprise finding was the difficulty of temperature control during the GPU training run. Both the graphics card and the processor temperatures were climbing steadily with the GPU getting dangerously close to 98 Celsius. Heat damage is a real system threat when training Tensorflow on the GPU.
Template portability needs work. Ideally, the entire project should be compiled into a Docker image (with docker-compose instructions). Environment variables should control directory structures, run times, and the variables should propagate all the way into application controls and outputs. Tensorflow models should also swap with minimal effort. Finally, R seems to be getting consumed by Python, so it would not hurt to add a Bokeh application as an R Shiny alternative.
Click below to see live demo of the final R Shiny App: