国产精品久久久aaaa,日日干夜夜操天天插,亚洲乱熟女香蕉一区二区三区少妇,99精品国产高清一区二区三区,国产成人精品一区二区色戒,久久久国产精品成人免费,亚洲精品毛片久久久久,99久久婷婷国产综合精品电影,国产一区二区三区任你鲁

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

有關 Dataflow 指令的原理

電子設計 ? 來源:電子設計 ? 作者:電子設計 ? 2022-02-09 10:34 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

本文轉載自:XILINX開發者社區微信公眾號

提取實現任務級 (task_level) 的硬件并行算法是設計高效的HLS IP內核的關鍵。

在本文中,我們將重點放在如何能夠在不需要特殊的庫或類的情況下修改代碼風格以實現C代碼實現并行性。Xilinx HLS 編譯器的顯著特征是能夠將任務級別的并行性和流水線與可尋址的存儲器 PIPO或 FIFO相結合。本文首先概述可以獲取任務并行的前提條件,然后以DAG(directedacyclic graph) 代碼為例,挖掘其中使用 fork-join 并行性,并結合使用 ping- pong buffer 啟用了一種基于握手的任務級粗粒度的流水線形式。

我們理解任務級并行的時候可以想象成這樣一個場景,每一個計算任務都是時間軸上向前奔跑的馬車,馬車與馬車之間傳輸的貨物就像是計算數據,他們需要管道去連接即 FIFO 和 PIPO ,FIFO 是一個先進先出存儲器也就是說使用這樣的管道傳輸數據的時候,數據進出的順序不可以改變。而 PIPO 就是一個可尋址的存儲器管道,數據在任務之間進出的順序可以改變。

最糟糕的狀態是什么?馬車在時間線上順序出發,A 馬車到達終點后 B 再出發以此類推,就像是 CPU 中的單進程順序執行模式一樣,而FPGA中有可供并行化執行的數據傳輸管道,更多的資源就像是跑道一樣,所以這個狀態效率是最低的。

那么先做一點點改進,我們分析發現 B 和 C 馬車不享有任何公用的數據或存儲計算資源,也就是他們完全可以在 A 結束后并行執行,最后再執行 D,這種并行情況中含有順序和并行兩種模式,我們稱之為交叉并行 (fork-joinparallelism)。 但是下一次進程仍然是順序執行的。

繼續深入可以發現,四輛馬車在跑完各自的任務后都有一段的閑置時間,提高吞吐量和資源重復利用也很明顯是息息相關的。實現了進程之間的流水線執行的結果就如下圖,每一輛馬車在不同的進程中連續執行任務,向前奔跑,重復利用資源的同時它提升了吞吐量進而極大的減小了完成多個進程后的延遲。

最理想的狀態時什么?就是馬車盡可能的一個挨著一個一起出發,并行奔跑,大家先后到達終點完成計算,在奔跑的過程中數據通過管道也完成了遷移,最終計算完的數據在最后一輛馬車到達終點的時候產出。下圖我們可以看到 B 和 C 開始執行的時間提前了,并沒有等到A完全執行完畢,這和數據依賴息息相關,也就是說我們進一步挖掘并行性的路上發現:ABC 三輛馬車都可以在增加馬車數量 (擴增資源) ,建立數據管道的并行執行的前提下實現了。我們用資源換取了更大的并行性,這就是繼續挖掘并行性上需要付出的代價。

奔跑的馬車帶著我們理解了任務級流水線的優化之路,下面我們結合代碼看一看HLS工具會在哪些情況下阻止 dataflow 的實現。

在我們談及 dataflow 的優化之前,我們先去了解在 HLS 提醒你報錯的方式,其中修改屬性config_dataflow-strict_mode (off | error | warning) 指令可以控制報錯指令的級別,一般情況下默認是 warning 級別的報錯,主要看我們的并行性需求。

以下是阻止任務級別并行性的常見情況:

1. 單產出單消耗模型違例(Single-producer-consumerviolations)

為了使 VitisHLS 執行 DATAFLOW 優化,任務之間傳遞的所有元素都必須遵循單產出單消耗模型。每個變量必須從單個任務驅動,并且只能由單個任務使用。在下面的代碼示例中是典型的單產出單消耗模型違例,單一的數據流 temp1 同時被 Loop2 和 Loop3 消耗。要解決這個問題很容易,就是將兩個任務都要消耗的數據流復制成兩個,如右圖的 Split 函數。當 temp1數據流被復制為 temp2 和 temp3 后,LOOP1,2,3 就可以實現任務級流水線了。

void foo(int data_in[N], int scale, int data_out1[N], int data_out2[N]) { int temp1[N]; Loop1: for(int i = 0; i < N; i++) { temp1[i] = data_in[i] * scale; } Loop2: for(int j = 0; j < N; j++) { data_out1[j] = temp1[j] * 123; } Loop3: for(int k = 0; k < N; k++) { data_out2[k] = temp1[k] * 456; } }void Split (in[N], out1[N], out2[N]) { // Duplicated data L1:for(int i=1;i

2. 旁路任務 Bypassing Tasks

正常情況下我們期望流水線任務是一個接著一個的產出并消耗,然而像下面這個例子中,Loop1 產生了 Temp1和Temp2 兩個數據流,但是在下一個任務 Loop2 中只有 temp1 參與了運算,而 temp2 就被旁支了。Loop3 任務的執行依賴 Loop2 任務產生的 temp3 數據,所以 Loop2 和 Loop3 因為數據依賴的關系無法并行執行。

void foo(int data_in[N], int scale, int data_out1[N], int data_out2[N]) { int temp1[N], temp2[N]. temp3[N]; Loop1: for(int i = 0; i < N; i++) { temp1[i] = data_in[i] * scale; temp2[i] = data_in[i] >> scale; } Loop2: for(int j = 0; j < N; j++) { temp3[j] = temp1[j] + 123; } Loop3: for(int k = 0; k

3. 任務間雙向反饋 Feedbackbetween Tasks

假如說當前任務的結果,需要作為之前一個任務的輸入的話,就形成了任務之間的數據反饋,它打亂了流水線從上級一直往下級輸送數據流的規則。這時候 HLS 就會給出警告或者報錯,有可能完成不了 dataflow 優化了。有一種特例是支持的:使用 hls::stream 格式的數據流反饋。

我們分析以下代碼的內容:

當第一個程序 firstProc 執行的時候,hls::stream 格式的數據流 forwardOUT 被寫入了初始化為10的數值 fromSecond 。由于 hls::stream 格式的數據本身不支持初始化操作,所以這樣的操作避免了違反單產出單消耗原則。之后的迭代里,firstProc 通過 backwardIN 接口從 hls :: stream 讀取數值寫入 forwardOUT 中。

在第二個程序 secondProc 執行的時候,secondProc 讀取 forwardIN 上的值,將其加1,然后通過按執行順序倒退的反饋流將其發送回 FirstProc。從第二次執行開始,firstProc 將使用從流中讀取的值進行計算,并且兩個過程可以使用第一次執行的初始值,通過正向和反饋通信永遠保持下去。這種交互式的反饋中,包含數據流的雙向反饋機制,但是它就像貨物一直在從左手倒到右手再從右手倒到左手一樣,可以不違反 Dataflow 的規范,一直進行下去。

#include "ap_axi_sdata.h" #include "hls_stream.h" void firstProc(hls::stream &forwardOUT, hls::stream &backwardIN) { static bool first = true; int fromSecond; //Initialize stream if (first) fromSecond = 10; // Initial stream value else //Read from stream fromSecond = backwardIN.read(); //Feedback value first = false; //Write to stream forwardOUT.write(fromSecond*2); } void secondProc(hls::stream &forwardIN, hls::stream &backwardOUT) { backwardOUT.write(forwardIN.read() + 1); } void top(...) { #pragma HLS dataflow hls::stream forward, backward; firstProc(forward, backward); secondProc(forward, backward); }

4. 含有條件判斷的任務流水

DATAFLOW 優化不會優化有條件執行的任務。下面的示例展現了這個違例。在此示例中,有條件地執行 Loop1 和 Loop2 會阻止 Vitis HLS 優化這些循環之間的數據流,因為 sel 條件直接控制了任務中的數據有可能不會從一個循環流到下一個循環。

void foo(int data_in1[N], int data_out[N], int sel) { int temp1[N], temp2[N]; if (sel) { Loop1: for(int i = 0; i < N; i++) { temp1[i] = data_in[i] * 123; temp2[i] = data_in[i]; } } else { Loop2: for(int j = 0; j < N; j++) { temp1[j] = data_in[j] * 321; temp2[j] = data_in[j]; } } Loop3: for(int k = 0; k < N; k++) { data_out[k] = temp1[k] * temp2[k]; } }

但是我們都知道,其實這些任務之間存在條件判斷和選擇是非常常見的情況,只需要稍微改變代碼風格就可以既保留條件判斷,又完成任務流水。為了確保在所有情況下都執行每個循環,我們將條件語句下變化的 Temp1 移入第一個循環。這兩個循環始終執行,并且數據始終從一個循環流向下一個循環。

void foo(int data_in[N], int data_out[N], int sel) { int temp1[N], temp2[N]; Loop1: for(int i = 0; i < N; i++) { if (sel) { temp1[i] = data_in[i] * 123; } else { temp1[i] = data_in[i] * 321; } } Loop2: for(int j = 0; j < N; j++) { temp2[j] = data_in[j]; } Loop3: for(int k = 0; k < N; k++) { data_out[k] = temp1[k] * temp2[k]; } }

5. 有多種退出機制的循環

含有多種退出機制的循環不能被包含在流水線區域內,我們來數一數 Loop2 一共有多少種循環退出條件:

1. 由 for 循環定義的 K>N 的情況;

2. 由 switch 條件定義的 default 情況;

3. 由 switch 條件定義的 continue 情況

由于循環的退出條件始終由循環邊界定義,因此使用 break 或 continue 語句將禁止在DATAFLOW 區域中使用循環。

void multi_exit(din_t data_in[N], dsc_t scale, dsel_t select, dout_t data_out[N]) { dout_t temp1[N], temp2[N]; int i,k; Loop1: for(i = 0; i < N; i++) { temp1[i] = data_in[i] * scale; temp2[i] = data_in[i] >> scale; } Loop2: for(k = 0; k < N; k++) { switch(select) { case 0: data_out[k] = temp1[k] + temp2[k]; case 1: continue; default: break; } } }

我們理解了可能阻止任務流水線的 5 種經典情況后,我們最后推出適用于 Vitis HLS 的Dataflow 優化的兩種規范形式 (canonical forms) ,一種直接應用于函數,一種應用于 for循環。我們可以發現規范形式嚴格遵守了單產出單消耗的規則。

1. 適用于子程序沒有被內聯 (inline) 的規范形式

void dataflow(Input0, Input1, Output0, Output1) { #pragma HLS dataflow UserDataType C0, C1, C2; func1(read Input0, read Input1, write C0, write C1); func2(read C0, read C1, write C2); func3(read C2, write Output0, write Output1); }

2. 適用于循環體內的任務流水的規范形式:

對于 for 循環 (其中沒有內聯函數的地方),循環變量應具有:

a. 在 for 循環的標題中聲明初始值,并設置為 0。

b. 循環條件N是一個正數值常數或常數函數參數。

c. 循環的遞增量為1。

d. Dataflow 指令必須位于循環內部。

void dataflow(Input0, Input1, Output0, Output1) { for (int i = 0; i < N; i++) { #pragma HLS dataflow UserDataType C0, C1, C2; func1(read Input0, read Input1, write C0, write C1); func2(read C0, read C0, read C1, write C2); func3(read C2, write Output0, write Output1); } }

有關 Dataflow 指令的原理,設計準則和規范形式都在本文講解給大家了,更多設計例程可以參考Github(https://github.com/Xilinx/HLS-Tiny-Tutorials/tree/master/coding_dataflow... ),如有疑問歡迎交流!

審核編輯:何安

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Dataflow
    +關注

    關注

    0

    文章

    4

    瀏覽量

    6892
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    A64指令集通關筆記:加載與存儲指令全解析

    作為嵌入式 ?Linux? 開發者, A64? 指令集是我們繞不開的基本功。最近我在復習這部分內容時,整理了一份帶思考題解答的筆記,希望能幫大家快速掌握核心要點。 ? 開篇:為什么必須啃下 A64
    的頭像 發表于 01-20 16:23 ?210次閱讀
    A64<b class='flag-5'>指令</b>集通關筆記:加載與存儲<b class='flag-5'>指令</b>全解析

    RISC-V高級指令融合如何實現性能密度躍升

    Synopsys ARC-V 處理器中的高級指令融合引入了一種新型機制,用于融合常見的 RISC-V 指令對,旨在提高處理器流水線效率,特別是針對資源受限的嵌入式處理器。通過融合來自不同功能單元的指令, 它將單發射、順序執行的處
    的頭像 發表于 12-29 11:22 ?326次閱讀
    RISC-V高級<b class='flag-5'>指令</b>融合如何實現性能密度躍升

    RDMA設計10:指令控制單元設計?

    指令控制單元負責控制數據傳輸及 DMA 數據讀寫。該單元簡化了數據傳輸及DMA 操作,使得用戶可以更簡便的控制數據傳輸事務。指令控制單元的功能由指令控制寄存器組和指令狀態寄存器組成。
    的頭像 發表于 12-23 09:24 ?304次閱讀
    RDMA設計10:<b class='flag-5'>指令</b>控制單元設計?

    歐盟CE-RED指令介紹

    歐盟CE-RED指令,全稱是RadioEquipmentDirective(無線電設備指令),指令編號為2014/53/EU,是歐盟針對所有帶無線發射功能設備制定的一項強制性法規。它是歐盟CE認證
    的頭像 發表于 11-05 14:01 ?718次閱讀
    歐盟CE-RED<b class='flag-5'>指令</b>介紹

    浮點數指令的添加——長指令寫回仲裁

    =在增加浮點數指令時,我們會遇到一些需要寫回寄存器的指令,此時就需要對原先的寫回功能模塊做更改。 寫回功能主要集中在這兩個模塊中 e203_exu_longpwbck.v長指令寫回仲裁
    發表于 10-24 07:09

    浮點數指令添加——長指令寫回仲裁

    浮點數指令添加——長指令寫回仲裁 在增加浮點數指令時,我們會遇到一些需要寫回寄存器的指令,此時就需要對原先的寫回功能模塊做更改。 寫回功能主要集中在這兩個模塊中
    發表于 10-24 06:07

    NICE指令的完整執行過程

    NICE指令的完整執行過程如下: 1、主處理器的譯碼單元提供EXU級譯碼得到指令的操作碼,以判斷其是否屬于默認的自定義指令組。 如果該指令屬于自定義
    發表于 10-23 07:25

    浮點指令擴展中16位指令的處理

    RISC-V支持16位壓縮格式,壓縮格式立即數位數更少,能使用的寄存器也比較少,有些指令只能用常用8個整數寄存器(x8-x15)或者(f8-f15)。16 位指令只對匯編器和鏈接器可見,并且是否以短
    發表于 10-20 11:02

    Linux環境下div指令、divu指令測試異常怎么解決?

    由于團隊修改了muldiv指令,需要測試div指令、divu指令是否功能正常: rv32um-p-div: 意外的,程序會產生自檢報錯(實際上這個報錯在原工程中也會出現),通過info我們可以看到
    發表于 10-20 06:07

    西門子SMART200脈沖指令資料

    西門子SMART200脈沖指令
    發表于 07-29 17:29 ?0次下載

    通用MCU語音芯片支持串口+AT指令485通訊

    文檔簡單介紹AD24N的性能參數、串口通訊指令、應用方向、等內容
    的頭像 發表于 07-24 11:35 ?708次閱讀
    通用MCU語音芯片支持串口+AT<b class='flag-5'>指令</b>485通訊

    步進電機控制指令詳解

    這些指令主要是針對用 PLC 直接聯動伺服放大器,目的是可以不借助其他擴展設備(例如1GM 模塊)來進行簡單的點位控制, 使用這些指令時最好配合三菱的伺服放大器 (如 MR-J2)。然而,我們也可以用這些指令來控制步進電機的運行
    的頭像 發表于 06-12 10:33 ?1996次閱讀
    步進電機控制<b class='flag-5'>指令</b>詳解

    NVME控制器設計之指令控制

    指令控制模塊由一個指令信息緩存, 一個指令組裝狀態機和一個 ID 池組成。 指令信息緩存中存放著由系統控制模塊寫入的待處理指令信息;
    的頭像 發表于 04-24 10:22 ?827次閱讀
    NVME控制器設計之<b class='flag-5'>指令</b>控制

    普源示波器遠程控制SCPI指令的應用

    Commands for Programmable Instruments)指令作為標準化的控制語言,為普源示波器的遠程控制提供了強有力的支持。本文將深入探討普源示波器遠程控制SCPI指令的應用,包括其基本概念、常用指令、應用
    的頭像 發表于 04-22 15:55 ?1264次閱讀
    普源示波器遠程控制SCPI<b class='flag-5'>指令</b>的應用

    CPU的各種指令和執行流程

    在集成電路設計中,CPU的指令是指計算機中央處理單元(CPU)用來執行計算任務的基本操作指令集。這些指令是CPU能夠理解并執行的二進制代碼,它們在計算機內部由硬件控制并按順序執行,從而實現計算、控制
    的頭像 發表于 04-18 11:24 ?2629次閱讀