Skip to content

shizukachan/darknet-nnpack

Forked from https://github.com/digitalbrain79/darknet-nnpack/

Darknet with NNPACK

NNPACK was used to optimize Darknet without using a GPU. It is useful for embedded devices using ARM CPUs.

Idein's qmkl is also used to accelerate the SGEMM using the GPU. This is slower than NNPACK on NEON-capable devices, and primarily useful for ARM CPUs without NEON.

The NNPACK implementation in Darknet was improved to use transform-based convolution computation, allowing for 40%+ faster inference performance on non-initial frames. This is most useful for repeated inferences, ie. video, or if Darknet is left open to continue processing input instead of allowed to terminate after processing input.

Build Instructions

Log in to Raspberry Pi using SSH.
Install PeachPy and confu

sudo apt-get install python-pip
sudo pip install --upgrade git+https://github.com/Maratyszcza/PeachPy
sudo pip install --upgrade git+https://github.com/Maratyszcza/confu

Install Ninja

git clone https://github.com/ninja-build/ninja.git
cd ninja
git checkout release
./configure.py --bootstrap
export NINJA_PATH=$PWD

Install clang (I'm not sure why we need this, NNPACK doesn't use it unless you specifically target it).

sudo apt-get install clang

Install modified NNPACK

git clone https://github.com/shizukachan/NNPACK
cd NNPACK
confu setup

If you are compiling for the Pi Zero, run python ./configure.py --backend scalar, otherwise run python ./configure.py --backend auto It's also recommended to examine and edit https://github.com/digitalbrain79/NNPACK-darknet/blob/master/src/init.c#L215 to match your CPU architecture if you're on ARM, as the cache size detection code only works on x86.

Since none of the ARM CPUs have a L3, it's recommended to set L3 = L2 and set inclusive=false. This should lead to the L2 size being set equal to the L3 size.

Ironically, after some trial and error, I've found that setting L3 to an arbitrary 2MB seems to work pretty well.

$NINJA_PATH/ninja
bin/convolution-inference-smoketest
sudo cp -a lib/* /usr/lib/
sudo cp include/nnpack.h /usr/include/
sudo cp deps/pthreadpool/include/pthreadpool.h /usr/include/

If the convolution-inference-smoketest fails, you've probably hit a compiler bug and will have to change to Clang or an older version of GCC. You can skip the qmkl/qasm/qbin2hex steps if you aren't targeting the QPU.

Install qmkl

sudo apt-get install cmake
git clone https://github.com/Idein/qmkl.git
cd qmkl
cmake .
make
sudo make install

Install qasm2

sudo apt-get install flex
git clone https://github.com/Terminus-IMRC/qpu-assembler2
cd qpu-assembler2
make
sudo make install

Install qbin2hex

git clone https://github.com/Terminus-IMRC/qpu-bin-to-hex
cd qpu-bin-to-hex
make
sudo make install

At this point, you can build darknet-nnpack using make. Be sure to edit the Makefile before compiling.

Test

The weight files can be downloaded from the YOLO homepage.

YOLOv2
./darknet detector test cfg/coco.data cfg/yolo.cfg yolo.weights data/person.jpg
Tiny-YOLO
./darknet detector test cfg/voc.data cfg/tiny-yolo-voc.cfg tiny-yolo-voc.weights data/person.jpg

Original NNPACK CPU-only Results (Raspberry Pi 3)

Model Build Options Prediction Time (seconds)
YOLOv2 NNPACK=1,ARM_NEON=1 8.2
YOLOv2 NNPACK=0,ARM_NEON=0 156
Tiny-YOLO NNPACK=1,ARM_NEON=1 1.3
Tiny-YOLO NNPACK=0,ARM_NEON=0 38

Improved NNPACK CPU-only Results (Raspberry Pi 3)

All NNPACK=1 results use march=native, pthreadpool is initialized for one thread for the single core Pi Zero, and mcpu=cortex-a53 for the Pi 3.

For non-implicit-GEMM convolution computation, it is possible to precompute the kernel to accelerate subsequent inferences. The first inference is slower than later ones, but the speedup is significant (40%+). This optimization is a classic time-memory tradeoff; YOLOv2 won't fit in the Raspberry Pi 3's memory with this code.

System Model Build Options Prediction Time (seconds)
Pi 3 YOLOv3-Tiny VOC NNPACK=1,ARM_NEON=1,NNPACK_FAST=1 1.1 (first frame), 0.73 (subsequent frames)
Pi 3 Tiny-YOLO NNPACK=1,ARM_NEON=1,NNPACK_FAST=1 1.4 (first frame), 0.82 (subsequent frames)
Pi 3 Tiny-YOLO NNPACK=1,ARM_NEON=1,NNPACK_FAST=0 1.2
Pi 3 Darknet 224x224 NNPACK=1,ARM_NEON=1,NNPACK_FAST=1 1.7 (first frame), 0.77 (subsequent frames)
Pi 3 Darknet 224x224 NNPACK=1,ARM_NEON=1,NNPACK_FAST=0 0.93
Pi 3 Darknet 256x256 NNPACK=1,ARM_NEON=1,NNPACK_FAST=1 1.8 (first frame), 0.87 (subsequent frames)
Pi 3 Darknet19 224x224 NNPACK=1,ARM_NEON=1,NNPACK_FAST=1 5.3 (first frame), 2.7 (subsequent frames)
Pi 3 Darknet19 224x224 NNPACK=1,ARM_NEON=1,NNPACK_FAST=0 3.8
Pi 3 Darknet19 256x256 NNPACK=1,ARM_NEON=1,NNPACK_FAST=1 5.8 (first frame), 3.1 (subsequent frames)
i5-3320M Tiny-YOLO NNPACK=1,NNPACK_FAST=1 0.27 (first frame), 0.17 (subsequent frames)
i5-3320M Tiny-YOLO NNPACK=1,NNPACK_FAST=0 0.42
i5-3320M Tiny-YOLO NNPACK=0, no OpenMP 1.4
i5-3320M YOLOv2 NNPACK=1,NNPACK_FAST=1 0.98 (first frame), 0.69 (subsequent frames)
i5-3320M YOLOv2 NNPACK=1,NNPACK_FAST=0 1.4
i5-3320M YOLOv2 NNPACK=0, no OpenMP 5.5

Apparently cfg files have changed with yolov3 update, so benchmarks may be out of date, ie. classifier network input size. This has been updated for the classifier networks Darknet and Darknet19 only.

On the Intel chip, using transformed GEMM is always faster, even with precomputation on the first frame, than implicit-GEMM. On the Pi 3, implicit-GEMM is faster on the first frame. This suggests that memory bandwidth may be a limiting factor on the Pi 3.

NNPACK+QPU_GEMM Results

I used these NNPACK cache tunings for the Pi 3:

L1 size: 32k / associativity: 4 / thread: 1
L2 size: 480k / associativity: 16 / thread: 4 / inclusive: false
L3 size: 2016k / associativity: 16 / thread: 1 / inclusive: false
This should yield l1.size=32, l2.size=120, and l3.size=2016 after NNPACK init is run.

And these for the Pi Zero:

L1 size: 16k / associativity: 4 / thread: 1
L2 size: 128k / associativity: 4 / thread: 1 / inclusive: false
L3 size: 128k / associativity: 4 / thread: 1 / inclusive: false
This should yield l1.size=16, l2.size=128, and l3.size=128 after NNPACK init is run.

Even though the Pi Zero's L2 is attached to the QPU and almost as slow as main memory, it does seem to have a small benefit.

Raspberry Pi Model Build Options Prediction Time (seconds)
Pi 3 Tiny-YOLO NNPACK=1,ARM_NEON=1,QPU_GEMM=1 5.3
Pi Zero Tiny-YOLO NNPACK=1,QPU_GEMM=1 7.7
Pi Zero Tiny-YOLO NNPACK=1,QPU_GEMM=0 28.2
Pi Zero Tiny-YOLO NNPACK=0,QPU_GEMM=0 124
Pi Zero Tiny-YOLO NNPACK=0,QPU_GEMM=1 8.0
Pi Zero Darknet19 224x224 NNPACK=1,QPU_GEMM=1 3.3
Pi Zero Darknet19 224x224 NNPACK=1,QPU_GEMM=0 22.3
Pi Zero Darknet19 224x224 NNPACK=0,QPU_GEMM=1 3.5
Pi Zero Darknet19 224x224 NNPACK=0,QPU_GEMM=0 96.3
Pi Zero Darknet 224x224 NNPACK=1,QPU_GEMM=1 1.23
Pi Zero Darknet 224x224 NNPACK=1,QPU_GEMM=0 4.15
Pi Zero Darknet 224x224 NNPACK=0,QPU_GEMM=1 1.32
Pi Zero Darknet 224x224 NNPACK=0,QPU_GEMM=0 14.9

On the Pi 3, the QPU is slower than NEON-NNPACK. qmkl is just unable to match the performance NNPACK's extremely well tuned NEON implicit GEMM.

On the Pi Zero, the QPU is faster than scalar-NNPACK. I have yet to investigate why enabling NNPACK gives a very slight speedup on the Pi Zero.

GPU / config.txt considerations

Using the QPU requires memory set aside for the GPU. Using the command sudo vcdbg reloc you can see how much memory is free on the GPU - it's roughly 20MB less than what is specified by gpu_mem.

I recommend no less than gpu_mem=80 if you want to run Tiny-YOLO/Darknet19/Darknet. The code I've used tries to keep GPU allocations to a minimum, but if Darknet crashes before GPU memory is freed, it will be gone until a reboot.