0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

TVM中schedule介紹

電子設(shè)計(jì) ? 來源:電子設(shè)計(jì) ? 作者:電子設(shè)計(jì) ? 2022-02-08 17:36 ? 次閱讀

作者:安平博,Xilinx高級工程師;來源:AI加速微信公眾號

Schedule是和硬件體系結(jié)構(gòu)相關(guān)的一些列優(yōu)化,Halide在其文章中對其做了以下定義:

1 When and where should be the value at each coordinate in each function be computed?

2 Where should they be stored?

3 How long are values cached and communicated across multiple consumers, and when are they independently recomputed by each?

第一條是描述了數(shù)據(jù)計(jì)算順序?qū)π阅艿挠绊?,第二條是數(shù)據(jù)的存儲位置對性能影響,最后一條是多線程處理過程中,不同線程數(shù)據(jù)應(yīng)該如何進(jìn)行交互。

參考文章:https://zhuanlan.zhihu.com/p/94846767,常用的shcedule有:

1 cache_read

將數(shù)據(jù)存儲到片上緩存,減少訪問數(shù)據(jù)時(shí)間。

2 cache_write

將結(jié)果寫入片上緩存,然后再寫入片外緩存。當(dāng)然這里的片上和片外并不是絕對的概念,也可以理解為不同層次的存儲結(jié)構(gòu)。

3 set_scope

為數(shù)據(jù)指定存儲位置,相比于cache_read和cache_write提供了更靈活的指定數(shù)據(jù)存儲方式。本質(zhì)上是相同的。

4 storage_align

在我看的文章中,storage_align是針對GPU shared memory的一個(gè)優(yōu)化,目的是為了減少同一個(gè)bank的訪問沖突。在GPU中shared memory被分割成多個(gè)bank,這些bank可以被獨(dú)立線程同時(shí)訪問。Storage_align就是為了將數(shù)據(jù)和bank大小匹配,減少bank conflict的發(fā)生。AI芯片中也有類似的問題,只有盡量減少bank沖突的發(fā)生,才能最大化并行計(jì)算。

5 compute_at

不懂CUDA,所以對文章中的代碼不是很理解,但是從其解釋看,對于多次循環(huán)的計(jì)算(或者多維計(jì)算),可以通過并行計(jì)算來降維。

6 compute_inline

將獨(dú)立操作轉(zhuǎn)化為內(nèi)聯(lián)函數(shù),有點(diǎn)類似FPGA上的流水線計(jì)算。轉(zhuǎn)化成內(nèi)聯(lián)函數(shù)從上層層面減少了stage。在FPGA中也有類似問題,可以將具有相同迭代的多條指令放在一起執(zhí)行。

7 compute_root

Compute_at的反操作。

8 fuse

將多個(gè)循環(huán)iter融合為一個(gè)iter。

9 split

Fuse的反操作,將一次循環(huán)迭代拆分為多次。

10 reorder

調(diào)整循環(huán)計(jì)算迭代順序。

11 tile

Tile也是將循環(huán)迭代進(jìn)行拆分,拆分多次計(jì)算。是split+reorder。

12 unroll

將循環(huán)展開,增加并發(fā)執(zhí)行。

13 vectorize

將循環(huán)迭代替換成ramp,可以通過SIMD指令實(shí)現(xiàn)數(shù)據(jù)批量計(jì)算,也就是單指令多數(shù)據(jù)計(jì)算。這在AI加速中會很常用,每條指令都是多數(shù)據(jù)計(jì)算的。

14 bind

CUDA中使用的優(yōu)化方法,將iter綁定到不同線程,實(shí)現(xiàn)并發(fā)計(jì)算。

15 parallel

實(shí)現(xiàn)多設(shè)備并行.

16 pragma

可以在代碼中人為添加編譯注釋,人為干預(yù)編譯優(yōu)化。HLS中就是通過這樣的方式來實(shí)現(xiàn)c的硬件編程的。

17 prefetch

將數(shù)據(jù)計(jì)算和load后者store數(shù)據(jù)重疊起來,在FPGA中是很常見優(yōu)化方法。

18 tensorize

將tensor作為一個(gè)整體匹配硬件的計(jì)算核心,比如一個(gè)卷積運(yùn)算就可以實(shí)現(xiàn)在FPGA上的一個(gè)匹配。

文章https://zhuanlan.zhihu.com/p/166551011 是通過官網(wǎng)的一個(gè)例子來介紹schedule的。在這個(gè)例子中,首先利用te的節(jié)點(diǎn)表達(dá)式建立了計(jì)算函數(shù),然后調(diào)用create_schedule來創(chuàng)建schedule實(shí)例,然后再調(diào)用lower函數(shù)實(shí)現(xiàn)schedule優(yōu)化。代碼如下:

# declare a matrix element-wise multiply A = te.placeholder((m, n), nam) B = te.placeholder((m, n), nam) C = te.compute((m, n), lambda i, j: A[i, j] * B[i, j], nam) s = te.create_schedule([C.op]) # lower will transform the computation from definition to the real # callable function. With argument `simple_mode=True`, it will # return you a readable C like statement, we use it here to print the # schedule result. print(tvm.lower(s, [A, B, C], simple_mode=True))

我這里依然延續(xù)上一章的內(nèi)容,看代碼中關(guān)于schedule的處理。

在上一章我們在codegen生成中,通過以下調(diào)用鏈轉(zhuǎn)到了schedule的處理。Codegen -> VisitExpr(CallNode* op) -> relay.backend._CompileEngineLower -> LowerInternal。LowerInternal函數(shù)為:

o4YBAGAJhSOAPhF1AAF7w2uy4BE784.png

如果是外部定義的編譯器,就只是建立cache_node節(jié)點(diǎn)和cache_func。如果是使用內(nèi)部編譯器,就會調(diào)用CreateSchedule建立schedule。接下來調(diào)用鏈為CreateSchedule -> ScheduleGetter.create -> te::create_schedule -> Schedule。create_schedule函數(shù)調(diào)用在文件re/schedule.h和te/schedule_lang.cc中。

create_schedule中主要有兩件工作:

1 創(chuàng)建ReadGraph,獲取post-dfs順序的算符圖。

2 初始化stage。

TVM中引入了stage的概念,一個(gè)op相當(dāng)于一個(gè)stage,schedule優(yōu)化是對stage的一個(gè)更改,可以增加,刪減,更改其特性等。

pIYBAGAJhWKALtoXAAExsOmziHM113.png

通過createReadGraph可以遍歷op圖,返回op和其依賴的tensor列表。和遍歷有關(guān)的主要函數(shù)為:

Op -> InputTensors -> PostOrderVisit -> IRApplyVisit,在IRApplyVisit中定義了VisitExpr和VisitStmt函數(shù)用于遍歷節(jié)點(diǎn)。

pIYBAGAJhaOAPU1EAAOCUW3ZpIQ102.png

Stmt節(jié)點(diǎn)通常是節(jié)點(diǎn)中的主體實(shí)現(xiàn),PrimExpr是TIR中節(jié)點(diǎn)的一個(gè)簡單表達(dá)式。比如if節(jié)點(diǎn):

o4YBAGAJheSAGUfwAAIKQRN_Atw687.png

ReadGraph創(chuàng)建完成后,通過PostDFSOrder來獲取post-dfs列表,其函數(shù)具體實(shí)現(xiàn)在graph.cc中,

o4YBAGAJhiSARaKbAAKhr5FEruU905.png

通過不斷迭代來進(jìn)行深度優(yōu)先搜索。

接下來是對stage進(jìn)行初始化。

首先對postorder中的所有op初始化一個(gè)stage對象。我們看以下stage的定義:

Stage類中主要定義了set_scope, compute_at, compute_root, bind, split, fuse等幾種優(yōu)化算法。同時(shí)定義了StageNode,在StageNode中定義了和優(yōu)化相關(guān)的變量,包括op,iter變量等??匆幌聅tage初始化代碼:

o4YBAGAJhmOAewapAAHNRZ9z5nY829.png

關(guān)鍵的幾個(gè)變量lef_iter_vars,all_iter_vars,這些有什么作用還需要深入看優(yōu)化函數(shù)的代碼。我們看幾個(gè)schedule函數(shù),先看一個(gè)最簡單的:compute_inline。代碼只有一行:

(*this)->attach_type = kInline

對于標(biāo)記了kInline的節(jié)點(diǎn),在lower的時(shí)候會進(jìn)行處理。應(yīng)該會將其直接和調(diào)用的節(jié)點(diǎn)結(jié)合,合并兩個(gè)op。

再看fuse函數(shù),其代碼為:

pIYBAGAJhqSAEGcxAAMXAmXWJb8660.png

IterVar表示計(jì)算中坐標(biāo)軸,比如一個(gè)兩級循環(huán),每級循環(huán)就是一個(gè)axis。從代碼中看出,fuse函數(shù)會對輸入的所有axis進(jìn)行合并,用fused變量替換合并后的axis。

這塊代碼比較抽象,先熟悉以下流程,之后再深入讀一下。

審核編輯:何安

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • TVM
    TVM
    +關(guān)注

    關(guān)注

    0

    文章

    19

    瀏覽量

    3642
收藏 人收藏

    評論

    相關(guān)推薦

    SemiDrive X9 AI 開發(fā)環(huán)境搭建

    SemiDrivex9AI開發(fā)環(huán)境搭建分開發(fā)機(jī)端,開發(fā)板端。主要的工具是SDNN,它是一個(gè)基于開源編譯器框架TVM的端到端的AI編譯器框架,Semidrive對TVM編譯器框架做了適配,主要特性如下
    的頭像 發(fā)表于 08-03 08:27 ?193次閱讀
    SemiDrive X9 AI 開發(fā)環(huán)境搭建

    使用TVM量化部署模型報(bào)錯(cuò)NameError: name \'GenerateESPConstants\' is not defined如何解決?

    各位好,我在使用TVM部署模型時(shí),遇到一下錯(cuò)誤,請問如何解決?我進(jìn)esp.py文件看,有如下兩個(gè)函數(shù)是找不到定義的: GenerateESPConstants(), ExtractConstantsFromPartitionedFunction(),
    發(fā)表于 06-28 10:50

    在e2 studio安裝QE的流程介紹

    在e2 studio安裝QE的流程介紹
    的頭像 發(fā)表于 04-04 08:05 ?253次閱讀
    在e2 studio<b class='flag-5'>中</b>安裝QE的流程<b class='flag-5'>介紹</b>

    RT-Thread程序在執(zhí)行rt_schedule_remove_thread(to_thread);語句里面的打印時(shí)會進(jìn)Hard_defaut的原因?

    ulog開啟 log_LVL等級 LOG_LVL_DBG log_Out_LVL等級 LOG_LVL_DBG 程序在執(zhí)行 rt_schedule_remove_thread(to_thread
    發(fā)表于 02-26 06:00

    AT89S51文資料介紹

    電子發(fā)燒友網(wǎng)站提供《AT89S51文資料介紹.pdf》資料免費(fèi)下載
    發(fā)表于 02-20 09:24 ?2次下載

    電源設(shè)計(jì)電容的工作原理介紹

    電源設(shè)計(jì),電容是一種非常重要的電子元件,它在電路起到濾波、耦合、儲能等作用。本文將對電源設(shè)計(jì)電容的工作原理進(jìn)行詳細(xì)介紹。 電容是一種能夠儲存電荷的電子元件,其基本結(jié)構(gòu)是由兩個(gè)相互
    的頭像 發(fā)表于 01-09 17:04 ?642次閱讀
    電源設(shè)計(jì)<b class='flag-5'>中</b>電容的工作原理<b class='flag-5'>介紹</b>

    TVM編譯器的整體架構(gòu)和基本方法

    。但是這其中也去思考了一下基于FPGA加速器的編譯器架構(gòu)。在FPGA深度學(xué)習(xí)加速器,編譯器除了需要自動化生成指令外,還要優(yōu)化指令的結(jié)構(gòu),來最大化加速器性能。TVM是一個(gè)支持GPU、CPU、FPGA指令
    的頭像 發(fā)表于 11-30 09:36 ?1863次閱讀
    <b class='flag-5'>TVM</b>編譯器的整體架構(gòu)和基本方法

    汽車EE體系結(jié)構(gòu)的功能域介紹

    汽車EE體系結(jié)構(gòu)的功能域介紹
    發(fā)表于 11-15 11:37 ?290次閱讀

    開關(guān)電源的硬開關(guān)和軟開關(guān)介紹

    電子發(fā)燒友網(wǎng)站提供《開關(guān)電源的硬開關(guān)和軟開關(guān)介紹.doc》資料免費(fèi)下載
    發(fā)表于 11-14 09:49 ?1次下載
    開關(guān)電源<b class='flag-5'>中</b>的硬開關(guān)和軟開關(guān)<b class='flag-5'>介紹</b>

    LabVIEWNIPackageManager功能介紹

    LabVIEWPackageManager功能介紹 使用NIPackage Manager可安裝、更新、修復(fù)和刪除NI軟件。 安裝NI軟件使用PackageManager瀏覽和安裝NI軟件
    發(fā)表于 11-13 18:59

    Schedule:簡單實(shí)用的 Python 周期任務(wù)調(diào)度工具

    如果你想在Linux服務(wù)器上周期性地執(zhí)行某個(gè) Python 腳本,最出名的選擇應(yīng)該是 Crontab 腳本,但是 Crontab 具有以下缺點(diǎn): ** 1.不方便執(zhí)行 秒級的任務(wù) 。 ** ** 2.當(dāng)需要執(zhí)行的定時(shí)任務(wù)有上百個(gè)的時(shí)候,Crontab的 管理就會特別不方便 。 ** 另外一個(gè)選擇是 Celery,但是 Celery 的配置比較麻煩,如果你只是需要一個(gè)輕量級的調(diào)度工具,Celery 不會是一個(gè)好選擇。 在你想要使用一個(gè)輕量級的任務(wù)調(diào)度工具,而且希望它盡量簡單、容易使用、不需要外部依賴,最好能夠容
    的頭像 發(fā)表于 10-30 11:18 ?575次閱讀

    IAR調(diào)試不同復(fù)位類型的介紹

    IAR調(diào)試不同復(fù)位類型的介紹在IAR 環(huán)境下調(diào)試有不同的復(fù)位類型,其中一些只復(fù)位內(nèi)核不復(fù)位MCU 外設(shè)的復(fù)位方式在特定情況下可能會造成仿真問題
    發(fā)表于 10-19 07:54

    高頻電路的電阻衰減網(wǎng)絡(luò)介紹

    介紹衰減網(wǎng)絡(luò)前,先介紹一下真實(shí)的電阻、電容、電感模型(1--5)
    的頭像 發(fā)表于 10-12 15:26 ?1778次閱讀
    高頻電路<b class='flag-5'>中</b>的電阻衰減網(wǎng)絡(luò)<b class='flag-5'>介紹</b>

    Linux為什么中斷不允許休眠

    Linux 為什么中斷不允許休眠? 所謂的睡眠,就是調(diào)用 schedule 讓出 CPU,調(diào)度器選擇另外個(gè)進(jìn)程繼續(xù)執(zhí)行,這個(gè)過程涉及進(jìn)程棧空間的切換。 1、假如中斷上下文中調(diào)用 schedule
    的頭像 發(fā)表于 09-27 17:48 ?1054次閱讀

    什么是LDO?LDO的噪聲和PSRR介紹

    在本文中,我們將介紹噪聲和電源抑制比 (PSRR) 在低壓差 (LDO) 穩(wěn)壓器的影響。讓我們簡要討論一下什么是 LDO。
    的頭像 發(fā)表于 09-26 14:29 ?3864次閱讀
    什么是LDO?LDO<b class='flag-5'>中</b>的噪聲和PSRR<b class='flag-5'>介紹</b>