This repository is a deployment project of BEVFormer on TensorRT, supporting FP32/FP16/INT8 inference. Meanwhile, in order to improve the inference speed of BEVFormer on TensorRT, this project implements some TensorRT Ops that support nv_half, nv_half2 and INT8. With the accuracy almost unaffected, the inference speed of the BEVFormer base can be increased by more than four times, the engine size can be reduced by more than 90%, and the GPU memory usage can be saved by more than 80%. In addition, the project also supports common 2D object detection models in MMDetection, which support INT8 Quantization and TensorRT Deployment with a small number of code changes.
Model | Data | Batch Size | NDS/mAP | FPS | Size (MB) | Memory (MB) | Device |
---|---|---|---|---|---|---|---|
BEVFormer tiny download |
NuScenes | 1 | NDS: 0.354 mAP: 0.252 |
15.9 | 383 | 2167 | RTX 3090 |
BEVFormer small download |
NuScenes | 1 | NDS: 0.478 mAP: 0.370 |
5.1 | 680 | 3147 | RTX 3090 |
BEVFormer base download |
NuScenes | 1 | NDS: 0.517 mAP: 0.416 |
2.4 | 265 | 5435 | RTX 3090 |
Model | Data | Batch Size | Float/Int | Quantization Method | NDS/mAP | FPS | Size (MB) | Memory (MB) | Device |
---|---|---|---|---|---|---|---|---|---|
BEVFormer tiny | NuScenes | 1 | FP32 | - | NDS: 0.354 mAP: 0.252 |
37.9 (x1) | 136 (x1) | 2159 (x1) | RTX 3090 |
BEVFormer tiny | NuScenes | 1 | FP16 | - | NDS: 0.354 mAP: 0.252 |
69.2 (x1.83) | 74 (x0.54) | 1729 (x0.80) | RTX 3090 |
BEVFormer tiny | NuScenes | 1 | FP32/INT8 | PTQ entropy per-tensor |
NDS: 0.353 mAP: 0.249 |
65.1 (x1.72) | 58 (x0.43) | 1737 (x0.80) | RTX 3090 |
BEVFormer tiny | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.353 mAP: 0.249 |
70.7 (x1.87) | 54 (x0.40) | 1665 (x0.77) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP32 | - | NDS: 0.478 mAP: 0.370 |
6.6 (x1) | 245 (x1) | 4663 (x1) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP16 | - | NDS: 0.478 mAP: 0.370 |
12.8 (x1.94) | 126 (x0.51) | 3719 (x0.80) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP32/INT8 | PTQ entropy per-tensor |
NDS: 0.476 mAP: 0.367 |
8.7 (x1.32) | 158 (x0.64) | 4079 (x0.87) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.477 mAP: 0.368 |
13.3 (x2.02) | 106 (x0.43) | 3441 (x0.74) | RTX 3090 |
BEVFormer base * | NuScenes | 1 | FP32 | - | NDS: 0.517 mAP: 0.416 |
1.5 (x1) | 1689 (x1) | 13893 (x1) | RTX 3090 |
BEVFormer base | NuScenes | 1 | FP16 | - | NDS: 0.517 mAP: 0.416 |
1.8 (x1.20) | 849 (x0.50) | 11865 (x0.85) | RTX 3090 |
BEVFormer base * | NuScenes | 1 | FP32/INT8 | PTQ entropy per-tensor |
NDS: 0.516 mAP: 0.414 |
1.8 (x1.20) | 426 (x0.25) | 12429 (x0.89) | RTX 3090 |
BEVFormer base * | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.515 mAP: 0.414 |
2.2 (x1.47) | 244 (x0.14) | 11011 (x0.79) | RTX 3090 |
* Out of Memory
when onnx2trt with TensorRT-8.5.1.7, but they convert successfully with TensorRT-8.4.3.1. So the version of these engines is TensorRT-8.4.3.1.
FP16 Plugins with nv_half
Model | Data | Batch Size | Float/Int | Quantization Method | NDS/mAP | FPS/Improve | Size (MB) | Memory (MB) | Device |
---|---|---|---|---|---|---|---|---|---|
BEVFormer tiny | NuScenes | 1 | FP32 | - | NDS: 0.355 mAP: 0.252 |
39.2 (x1.03) | 135 (x0.99) | 1691 (x0.78) | RTX 3090 |
BEVFormer tiny | NuScenes | 1 | FP16 | - | NDS: 0.355 mAP: 0.251 |
79.6 (x2.10) | 72 (x0.53) | 1203 (x0.56) | RTX 3090 |
BEVFormer tiny | NuScenes | 1 | FP32/INT8 | PTQ entropy per-tensor |
NDS: 0.352 mAP: 0.249 |
88.3 (x2.33) | 59 (x0.43) | 1199 (x0.56) | RTX 3090 |
BEVFormer tiny | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.353 mAP: 0.250 |
97.5 (x2.57) | 52 (x0.38) | 1055 (x0.49) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP32 | - | NDS: 0.478 mAP: 0.370 |
7.1 (x1.08) | 250 (x1.02) | 2625 (x0.56) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP16 | - | NDS: 0.479 mAP: 0.371 |
15.4 (x2.33) | 124 (x0.51) | 1723 (x0.37) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP32/INT8 | PTQ entropy per-tensor |
NDS: 0.477 mAP: 0.368 |
16.7 (x2.53) | 165 (x0.67) | 1639 (x0.35) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.477 mAP: 0.367 |
18.7 (x2.83) | 109 (x0.44) | 1471 (x0.32) | RTX 3090 |
BEVFormer base | NuScenes | 1 | FP32 | - | NDS: 0.517 mAP: 0.416 |
2.9 (x1.93) | 292 (x0.17) | 5343 (x0.38) | RTX 3090 |
BEVFormer base | NuScenes | 1 | FP16 | - | NDS: 0.518 mAP: 0.416 |
5.1 (x3.40) | 148 (x0.09) | 3317 (x0.24) | RTX 3090 |
BEVFormer base | NuScenes | 1 | FP32/INT8 | PTQ entropy per-tensor |
NDS: 0.515 mAP: 0.413 |
6.3 (x4.20) | 201 (x0.12) | 3305 (x0.24) | RTX 3090 |
BEVFormer base | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.509 mAP: 0.408 |
7.2 (x4.80) | 132 (x0.08) | 2457 (x0.18) | RTX 3090 |
FP16 Plugins with nv_half2
Model | Data | Batch Size | Float/Int | Quantization Method | NDS/mAP | FPS | Size (MB) | Memory (MB) | Device |
---|---|---|---|---|---|---|---|---|---|
BEVFormer tiny | NuScenes | 1 | FP16 | - | NDS: 0.355 mAP: 0.252 |
90.7 (x2.39) | 72 (x0.53) | 1203 (x0.56) | RTX 3090 |
BEVFormer tiny | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.354 mAP: 0.250 |
101.7 (x2.68) | 54 (x0.40) | 1091 (x0.51) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP16 | - | NDS: 0.478 mAP: 0.370 |
17.9 (x2.71) | 152 (x0.62) | 1639 (x0.35) | RTX 3090 |
BEVFormer small | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.478 mAP: 0.369 |
21.5 (x3.26) | 134 (x0.55) | 1507 (x0.32) | RTX 3090 |
BEVFormer base | NuScenes | 1 | FP16 | - | NDS: 0.517 mAP: 0.416 |
6.6 (x4.40) | 172 (x0.10) | 3341 (x0.24) | RTX 3090 |
BEVFormer base | NuScenes | 1 | FP16/INT8 | PTQ entropy per-tensor |
NDS: 0.516 mAP: 0.414 |
7.7 (x5.13) | 133 (x0.08) | 2691 (x0.19) | RTX 3090 |
This project also supports common 2D object detection models in MMDetection with little modification. The following are deployment examples of YOLOx and CenterNet.
Model | Data | Framework | Batch Size | Float/Int | Quantization Method | mAP | FPS | Size (MB) | Memory (MB) | Device |
---|---|---|---|---|---|---|---|---|---|---|
YOLOx download |
COCO | PyTorch | 32 | FP32 | - | mAP: 0.506 | 63.1 | 379 | 7617 | RTX 3090 |
YOLOx | COCO | TensorRT | 32 | FP32 | - | mAP: 0.506 | 71.3 (x1) | 546 (x1) | 9943 (x1) | RTX 3090 |
YOLOx | COCO | TensorRT | 32 | FP16 | - | mAP: 0.506 | 296.8 (x4.16) | 192 (x0.35) | 4567 (x0.46) | RTX 3090 |
YOLOx | COCO | TensorRT | 32 | FP32/INT8 | PTQ entropy per-tensor |
mAP: 0.488 | 556.4 (x7.80) | 99 (x0.18) | 5225 (x0.53) | RTX 3090 |
YOLOx | COCO | TensorRT | 32 | FP16/INT8 | PTQ entropy per-tensor |
mAP: 0.479 | 550.6 (x7.72) | 99 (x0.18) | 5119 (x0.51) | RTX 3090 |
Model | Data | Framework | Batch Size | Float/Int | Quantization Method | mAP | FPS | Size (MB) | Memory (MB) | Device |
---|---|---|---|---|---|---|---|---|---|---|
CenterNet download |
COCO | PyTorch | 32 | FP32 | - | mAP: 0.299 | 337.4 | 56 | 5171 | RTX 3090 |
CenterNet | COCO | TensorRT | 32 | FP32 | - | mAP: 0.299 | 475.6 (x1) | 58 (x1) | 8241 (x1) | RTX 3090 |
CenterNet | COCO | TensorRT | 32 | FP16 | - | mAP: 0.297 | 1247.1 (x2.62) | 29 (x0.50) | 5183 (x0.63) | RTX 3090 |
CenterNet | COCO | TensorRT | 32 | FP32/INT8 | PTQ entropy per-tensor |
mAP: 0.27 | 1534.0 (x3.22) | 20 (x0.34) | 6549 (x0.79) | RTX 3090 |
CenterNet | COCO | TensorRT | 32 | FP16/INT8 | PTQ entropy per-tensor |
mAP: 0.285 | 1889.0 (x3.97) | 17 (x0.29) | 6453 (x0.78) | RTX 3090 |
git clone [email protected]:DerryHub/BEVFormer_tensorrt.git
cd BEVFormer_tensorrt
PROJECT_DIR=$(pwd)
Download the COCO 2017 datasets to /path/to/coco
and unzip them.
cd ${PROJECT_DIR}/data
ln -s /path/to/coco coco
Download nuScenes V1.0 full dataset data and CAN bus expansion data HERE as /path/to/nuscenes
and /path/to/can_bus
.
Prepare nuscenes data like BEVFormer.
cd ${PROJECT_DIR}/data
ln -s /path/to/nuscenes nuscenes
ln -s /path/to/can_bus can_bus
cd ${PROJECT_DIR}
sh samples/bevformer/create_data.sh
${PROJECT_DIR}/data/.
├── can_bus
│ ├── scene-0001_meta.json
│ ├── scene-0001_ms_imu.json
│ ├── scene-0001_pose.json
│ └── ...
├── coco
│ ├── annotations
│ ├── test2017
│ ├── train2017
│ └── val2017
└── nuscenes
├── maps
├── samples
├── sweeps
└── v1.0-trainval
Download and install the CUDA-11.6/cuDNN-8.6.0/TensorRT-8.5.1.7
following NVIDIA.
Install PyTorch and TorchVision following the official instructions.
pip install torch==1.12.1+cu116 torchvision==0.13.1+cu116 torchaudio==0.12.1+cu116 --extra-index-url https://download.pytorch.org/whl/cu116
git clone https://github.com/open-mmlab/mmcv.git
cd mmcv
git checkout v1.5.0
pip install -r requirements/optional.txt
MMCV_WITH_OPS=1 pip install -e .
git clone https://github.com/open-mmlab/mmdetection.git
cd mmdetection
git checkout v2.25.1
pip install -v -e .
# "-v" means verbose, or more output
# "-e" means installing a project in editable mode,
# thus any local modifications made to the code will take effect without reinstallation.
git clone [email protected]:open-mmlab/mmdeploy.git
cd mmdeploy
git checkout v0.10.0
git clone [email protected]:NVIDIA/cub.git third_party/cub
cd third_party/cub
git checkout c3cceac115
# go back to third_party directory and git clone pybind11
cd ..
git clone [email protected]:pybind/pybind11.git pybind11
cd pybind11
git checkout 70a58c5
Make sure cmake version >= 3.14.0 and gcc version >= 7.
export MMDEPLOY_DIR=/the/root/path/of/MMDeploy
export TENSORRT_DIR=/the/path/of/tensorrt
export CUDNN_DIR=/the/path/of/cuda
export LD_LIBRARY_PATH=$TENSORRT_DIR/lib:$LD_LIBRARY_PATH
export LD_LIBRARY_PATH=$CUDNN_DIR/lib64:$LD_LIBRARY_PATH
cd ${MMDEPLOY_DIR}
mkdir -p build
cd build
cmake -DCMAKE_CXX_COMPILER=g++-7 -DMMDEPLOY_TARGET_BACKENDS=trt -DTENSORRT_DIR=${TENSORRT_DIR} -DCUDNN_DIR=${CUDNN_DIR} ..
make -j$(nproc)
make install
cd ${MMDEPLOY_DIR}
pip install -v -e .
# "-v" means verbose, or more output
# "-e" means installing a project in editable mode,
# thus any local modifications made to the code will take effect without reinstallation.
cd ${PROJECT_DIR}/TensorRT/build
cmake .. -DCMAKE_TENSORRT_PATH=/path/to/TensorRT
make -j$(nproc)
make install
Run Unit Test of Custom TensorRT Plugins
cd ${PROJECT_DIR}
sh samples/test_trt_ops.sh
cd ${PROJECT_DIR}/third_party/bevformer
python setup.py build develop
Download above PyTorch checkpoints to ${PROJECT_DIR}/checkpoints/pytorch/
. The ONNX files and TensorRT engines will be saved in ${PROJECT_DIR}/checkpoints/onnx/
and ${PROJECT_DIR}/checkpoints/tensorrt/
.
Support Common TensorRT Ops in BEVFormer: Grid Sampler
, Multi-scale Deformable Attention
, Modulated Deformable Conv2d
and Rotate
.
Each operation is implemented as 2 versions: FP32/FP16 (nv_half)/INT8 and FP32/FP16 (nv_half2)/INT8.
For specific speed comparison, see Custom TensorRT Plugins.
The following tutorial uses BEVFormer base
as an example.
- Evaluate with PyTorch
cd ${PROJECT_DIR}
# defult gpu_id is 0
sh samples/bevformer/base/pth_evaluate.sh -d ${gpu_id}
- Evaluate with TensorRT and MMDeploy Plugins
# convert .pth to .onnx
sh samples/bevformer/base/pth2onnx.sh -d ${gpu_id}
# convert .onnx to TensorRT engine (FP32)
sh samples/bevformer/base/onnx2trt.sh -d ${gpu_id}
# convert .onnx to TensorRT engine (FP16)
sh samples/bevformer/base/onnx2trt_fp16.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP32)
sh samples/bevformer/base/trt_evaluate.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP16)
sh samples/bevformer/base/trt_evaluate_fp16.sh -d ${gpu_id}
# Quantization
# calibration and convert .onnx to TensorRT engine (FP32/INT8)
sh samples/bevformer/base/onnx2trt_int8.sh -d ${gpu_id}
# calibration and convert .onnx to TensorRT engine (FP16/INT8)
sh samples/bevformer/base/onnx2trt_int8_fp16.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP32/INT8)
sh samples/bevformer/base/trt_evaluate_int8.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP16/INT8)
sh samples/bevformer/base/trt_evaluate_int8_fp16.sh -d ${gpu_id}
# quantization aware train
# defult gpu_ids is 0,1,2,3,4,5,6,7
sh samples/bevformer/base/quant_aware_train.sh -d ${gpu_ids}
# then following the post training quantization process
- Evaluate with TensorRT and Custom Plugins
# nv_half
# convert .pth to .onnx
sh samples/bevformer/plugin/base/pth2onnx.sh -d ${gpu_id}
# convert .onnx to TensorRT engine (FP32)
sh samples/bevformer/plugin/base/onnx2trt.sh -d ${gpu_id}
# convert .onnx to TensorRT engine (FP16-nv_half)
sh samples/bevformer/plugin/base/onnx2trt_fp16.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP32)
sh samples/bevformer/plugin/base/trt_evaluate.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP16-nv_half)
sh samples/bevformer/plugin/base/trt_evaluate_fp16.sh -d ${gpu_id}
# nv_half2
# convert .pth to .onnx
sh samples/bevformer/plugin/base/pth2onnx_2.sh -d ${gpu_id}
# convert .onnx to TensorRT engine (FP16-nv_half2)
sh samples/bevformer/plugin/base/onnx2trt_fp16_2.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP16-nv_half2)
sh samples/bevformer/plugin/base/trt_evaluate_fp16_2.sh -d ${gpu_id}
# Quantization
# nv_half
# calibration and convert .onnx to TensorRT engine (FP32/INT8)
sh samples/bevformer/plugin/base/onnx2trt_int8.sh -d ${gpu_id}
# calibration and convert .onnx to TensorRT engine (FP16-nv_half/INT8)
sh samples/bevformer/plugin/base/onnx2trt_int8_fp16.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP32/INT8)
sh samples/bevformer/plugin/base/trt_evaluate_int8.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP16-nv_half/INT8)
sh samples/bevformer/plugin/base/trt_evaluate_int8_fp16.sh -d ${gpu_id}
# nv_half2
# calibration and convert .onnx to TensorRT engine (FP16-nv_half2/INT8)
sh samples/bevformer/plugin/base/onnx2trt_int8_fp16_2.sh -d ${gpu_id}
# evaluate with TensorRT engine (FP16-nv_half2/INT8)
sh samples/bevformer/plugin/base/trt_evaluate_int8_fp16_2.sh -d ${gpu_id}
This project is mainly based on these excellent open source projects: