云端深度學習的服務的性能加速通常需要算法和工程的協同加速,需要模型推理和計算節點的融合,并保證整個“木桶”沒有太明顯的短板。
如何在滿足時延前提下讓算法工程師的服務的吞吐盡可能高,盡可能簡便成了性能優化的關鍵一環。為了解決這些問題,TorchPipe通過深入PyTorch的C++計算后端和CUDA流管理,以及針對多節點的領域特定語言建模,對外提供面向PyTorch前端的線程安全函數接口,對內提供面向用戶的細粒度后端擴展。
背景
深度學習的Serving面臨多個難題:
一是GIL鎖帶來的多線程使用受限
三是復雜流程
業界有一些實踐,如triton inference server, 阿里媽媽high_service, 美團視覺GPU推理服務部署架構優化實踐。總體上,有以下方向去做這些事情:
全流程gpu化
DAG的并行化
對于cpu計算后端,去克服GIL鎖
通常用戶對于trinton inference server的一個抱怨是,在多個節點交織的系統中,有大量業務邏輯在客戶端完成,通過RPC調用服務端,很麻煩;而為了性能考慮,不得不考慮共享顯存,ensemble,BLS[5]等非常規手段。
一. 問題定義
對于我們自己來說,面臨的第一個問題是,pytorch 中如何并發調用resnet18模型。本項目開始于一個簡單的需求,即我們需要求得一個 X,能夠實現模型推理并滿足:
前向接口需要是線程安全的。
在主要硬件平臺(如 NVIDIA GPU)以及主要通用加速引擎(如 TensorRT/Libtorch)上實現了此 X。
import torch, X resnet18 = X(model="resnet18_-1x3x224x224.onnx", # 動態尺度 precision="fp16", max=4, # 動態batch模型的最大batch數目 instance_num=4, # 多個并行計算實例 batching_timeout=5) # 湊batch超時時間 data = torch.from_numpy(data) net_output: torch.Tensor = resnet18(data=data) # 線程安全調用
以此為起點,我們擴展到了對以下場景的支持:
包含前處理在內的通用計算后端X的細粒度泛型擴展
多節點組成的有向無環圖(DAG)的流水線并行,多級結構化
條件控制流
二. 背景知識
首先,我們介紹一些背景知識。您也可以跳過這一部分。
2.1 CUDA: 流和并發
CUDA提供了一致的抽象,來控制并發訪問,以便用戶最大化、完整地利用單塊GPU設備的資源能力。為了最有效的使用GPU設備,我們希望:
- 單位硬件資源能承載更多的業務請求量
- GPU盡可能滿載(前提:關聯資源使用量小,時延達標)
為了達到此目的,我們簡單分析下CUDA的編程模型。
硬件
Cuda Core是顯卡主要的運算單元。采用Pascal及以上架構的顯卡擁有上千的CUDA核心,對應著多組SM(Streaming Multiprocessor),可共用于一個任務,也可承載不同的運算任務。從volta架構開始,NVIDIA引入了專為深度學習設計的Tensor Core. 在Turing架構的 Tesla T4中,一共有40個SM, 共享6MB的L2緩存。一個SM由64個FP32 算數單元,和8個Tensor Core組成。對于模型的算子級優化,需要關注較為底層的優化。而對于業務使用場景,既需要算子級優化(選取針對性的計算后端負責),也需要整體視角的分析。
參考鏈接: GPU Architecture.
CUDA流
CUDA流表示一個GPU操作隊列,所有提交給GPU的任務,均指定了執行流。存在一個默認流,也就是`stream 0`, 作為默認的隊列。`提交任務`這個操作本身可以是異步的,對流進行同步化,則意味著需要阻塞cpu線程,直至所有已經提交至該隊列中的任務執行完畢。不同流之間的任務可以借助硬件的不同單元并行執行或者時分并發執行。
CUDA上下文(CUDA Context)
CUDA-Stream/CUDA-Context可以類比于線程/進程:多線程分配調用的GPU資源同屬一個CUDA Context下,有自己的隔離的地址空間,資源不能跨Context共享。默認情況下,一個進程中,在初次調用CUDA runtime軟件庫中的任何一個API時,會自動初始化當前進程中唯一的一個CUDA上下文。GPU在同一時刻只能切換到一個context,而默認情況下一個進程有一個上下文,故多個進程使用GPU,無法同時利用硬件。
虛擬化
由于GPU無法同時執行跨CUDA context的任務,導致硬件利用率可能不高,此時可采用一些虛擬化手段。典型的如NVIDIA官方的MPS(Multi Process Service),它實際上啟動了一個獨立進程去轉發所有的任務。采用此方法的壞處是隔離性收到了一定破壞:一旦此進程失效,所有關聯任務都將受到影響。
為了充分利用GPU的性能,可以采取一些措施:
- GPU任務合理分配到多個流,并只在恰當時機同步;
- 將單個顯卡的任務限制在單個進程中,去克服CUDA上下文分時特性帶來的資源利用率可能不足的問題。
2.2 PyTorch CUDA 語義
PyTorch 以易用性為核心,按照一致的原則組織了對GPU資源的訪問。
當前流
在PyTorch內,當前流(current stream)指的是當前線程綁定的CUDA流。PyTorch通過以下API提供了綁定CUDA流到當前線程,以及獲取當前線程綁定的CUDA流的功能:
torch.cuda.set_stream(stream) torch.cuda.current_stream(device=None)默認情況下,所有線程都綁定到默認流(stream 0)上. PyTorch的GPU運算均提交到當前線程綁定的`當前流`上。PyTorch盡量讓用戶感知不到這點: - 通常來說,當前流是都是默認流,而在同一個流上提交的任務會按提交時間串行執行; - 對于涉及到將GPU數據拷貝到CPU或者另外一塊GPU設備的操作, PyTorch默認地在操作中插入當前流的同步操作 . 為了在多線程環境使得PyTorch充分利用GPU資源,我們需要打破以上慣例:
計算后端線程綁定到獨立的CUDA流;
在線程轉換時進行流同步
三. 單節點的并行化
3.1 resnet18 計算加速
對于onnx格式的 resnet18的模型resnet18_-1x3x224x224.onnx, 通常有以下手段進行推理加速:
使用tensorrt等框架進行模型針對性加速
避免頻繁顯存申請
多實例,batching,分別用來提高資源使用量和使用效率
優化數據傳輸
線程安全的本地推理
為了方便,假設將tensorrt推理功能封裝為名稱為 TensorrtTensor 的計算后端。由于計算發生在gpu設備上,我們加上SyncTensor 表示gpu上的流同步操作。
配置項 | 參數 | 說明 |
backend | "SyncTensor[TensorrtTensor]" | 計算后端和tensorrt推理本身一樣,不是線程安全的。 |
max | 4 | 模型支持的最大batchsize,用于模型轉換(onnx->tensorrt) |
torchpipe默認會在此計算后端上包裹一層可擴展的單節點調度后端,實現以下三個基本能力:
前向接口線程安全性
多實例并行
配置項 | 默認值 | 說明 |
instance_num | 1 | 多個模型實例并行執行推理任務。 |
Batching
對于resnet18, 模型本身輸入為-1x3x224x224, batchsize越大,單位硬件資源所完成的任務越多。batchsize 從計算后端(TensorrtTensor)讀取。
配置項 | 默認值 | 說明 |
batching_timeout | 0 | 單位為毫秒,在此時間內如果沒有接收到 batchsize 個數目的請求,則放棄等待。 |
性能調優技巧
匯總以上步驟,我們獲得推理resnet18在torchpipe下的必要參數:
import torchpipe as tp import torch config = { # 單節點調度器參數: "instance_num":2, "batching_timeout":5, # 計算后端: "backend":"SyncTensor[TensorrtTensor]", # 計算后端參數: "model":"resnet18_-1x3x224x224.onnx", "max":4 } # 初始化 models = tp.pipe(config) data = torch.ones(1,3,224,224).cuda() ## 前向 input = {"data":data} models(input) # <== 可多線程調用 result: torch.Tensor = input["result"] # 失敗則 "result" 不存在
假設我們想要支持最多10路的客戶端/并發請求, instance_num 一般設置2,以便最多有處理 instance_num*max = 8 路的能力。
性能取舍 請注意,我們的加速做了如下假設: 同設備上的數據拷貝(如cpu-cpu數據拷貝,gpu-gpu同一顯卡內部顯存拷貝)速度快,消耗資源少,整體上可忽略不計。 相對于cpu-gpu數據拷貝以及其他的計算,這條假設是沒問題的。后面我們將看到,在一些特殊場景,這條假設可能不成立,需要相應的規避手段。
3.2 計算后端
在深度學習的服務中,如果僅支持模型加速遠遠不夠。為此,我們內置了一些常用的細粒度后端。
內置后端舉例:
名稱 | 說明 |
DecodeMat | jpg解碼 |
cvtColorMat | 顏色空間轉換 |
ResizeMat | resize |
PillowResizeMat | 嚴格保持和pillow的結果一致的resize |
更多... |
名稱 | 說明 |
DecodeTensor | GPU上jpg解碼 |
cvtColorTensor | 顏色空間轉換 |
ResizeTensor | resize |
PillowResizeTensor | 嚴格保持和pillow的結果一致的resize |
更多... |
3.3 Sequential
Sequential能串聯多個后端。也就是說,Sequential[DecodeTensor,ResizeTensor,cvtColorTensor,SyncTensor]和Sequential[DecodeMat,ResizeMat]是有效后端。
在Sequential[DecodeMat,ResizeMat]的前向執行中,數據(dict)會依次經過下列流程:
執行 DecodeMat:DecodeMat讀取data, 并將結果賦值給result和color
條件控制流:嘗試將數據中的result的值賦值給data 并刪除result
執行 ResizeMat :ResizeMat讀取data, 并將結果賦值給result鍵值
Sequential可簡寫為S.
3.4 單節點調度系統
輸入數據經由默認的單節點調度系統BaselineSchedule分發給計算后端執行。在此過程中主要經歷了湊batch和多實例的調度。
湊batch/多實例
對于TensorrtTensor等模型推理引擎,輸入范圍一般是[1, max_batch_size], 此時調度系統可將輸入數據打包送入。BaselineSchedule單節點調度后端實現了如下的調度功能:
根據instance_num參數啟動多個計算后端實例
從計算后端讀取max_batch_size=max(), 如果大于1,啟動湊batch功能
從輸入隊列獲取數據,在batching_timeout的時間內,如果獲得了max_batch_size個數據,那么將其送往Batch隊列, 如果時間到了仍然沒有獲得足夠數據,那么將已有數據送入Batch隊列
將任務從Batch隊列中分發到空閑的計算實例中。
以上是主干的大致流程,細節部分會有差別,如BaselineSchedule也實現了基礎的自適應流量功能,根據多實例計算引擎的狀態決定batch狀態的功能,以及組合調度的功能。
單節點組合調度
有些計算后端的輸入范圍最小值大于1, 導致無法作為正常的后端進行調度(可能導致有些數據永遠沒有辦法進行處理)。BaselineSchedule通過&符號提供了組合的能力。
舉例來講,對于TensorrtTensor后端,一些模型不方便轉為動態模型, 此時可以用一個 batchsize=1 的模型和幾個 batchsize=N 的模擬動態batch.
[model] model="batch1.onnx&batch4.onnx&batch8.onnx" backend="SyncTensor[TensorrtTensor]" # or 'SyncTensor[TensorrtTensor]&SyncTensor[TensorrtTensor]' instance_num = 2 # auto extend to '2&2&2' min="1&4&8" max="1&4&8"
此時,將共有6個實例,前兩個實例輸入范圍均是[1, 1],中間兩個均是[4, 4],最后兩個均是[8, 8]。對BaselineSchedule來說,這六個實例組成了兩個虛擬實例,每個虛擬實例占用了三個實例,虛擬實例的輸入范圍是[1, 8].
四. 多節點調度
針對多節點,主要考慮了:
多個節點的鏈接, filter: 有向無環圖中的條件控制流, context: 自動map語法糖, 圖的跳轉, 邏輯節點。
五. RoadMap
torchpie目前處于一個快速迭代階段,我們非常需要你的幫助。歡迎通過issues或者反饋等方式幫助我們。 我們的最終目標是讓服務端高吞吐部署盡可能簡單。為了實現這一目標,我們將積極自我迭代,也愿意參與有相近目標的其他項目。
2023年度和2024年度 RoadMap
大模型方面的示例
公開的基礎鏡像和pypi(manylinux)
優化編譯系統,分為core,pplcv,model/tensorrt,opencv等模塊
基礎結構優化。包含python與c++交互,異常,日志系統,跨進程后端的優化;
技術報告
潛在未完成的研究方向
單節點調度和多節點調度后端,他們與計算后端無本質差異,需要更多面向用戶進行解耦,我們想要將這部分優化為用戶API的一部分;
針對多節點的調試工具。由于在多節點調度中,使用了模擬棧設計,比較容易設計節點級別的調試工具;
負載均衡
審核編輯:湯梓紅
-
cpu
+關注
關注
68文章
10901瀏覽量
212766 -
多線程
+關注
關注
0文章
278瀏覽量
20057 -
C++
+關注
關注
22文章
2114瀏覽量
73813 -
pytorch
+關注
關注
2文章
808瀏覽量
13337
原文標題:torchpipe : Pytorch 內的多線程計算并行庫
文章出處:【微信號:GiantPandaCV,微信公眾號:GiantPandaCV】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論