一区二区三区三上|欧美在线视频五区|国产午夜无码在线观看视频|亚洲国产裸体网站|无码成年人影视|亚洲AV亚洲AV|成人开心激情五月|欧美性爱内射视频|超碰人人干人人上|一区二区无码三区亚洲人区久久精品

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

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

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

有關(guān) Dataflow 指令的原理

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

掃碼添加小助手

加入工程師交流群

本文轉(zhuǎn)載自:XILINX開發(fā)者社區(qū)微信公眾號(hào)

提取實(shí)現(xiàn)任務(wù)級(jí) (task_level) 的硬件并行算法是設(shè)計(jì)高效的HLS IP內(nèi)核的關(guān)鍵。

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

我們理解任務(wù)級(jí)并行的時(shí)候可以想象成這樣一個(gè)場景,每一個(gè)計(jì)算任務(wù)都是時(shí)間軸上向前奔跑的馬車,馬車與馬車之間傳輸?shù)呢浳锞拖袷怯?jì)算數(shù)據(jù),他們需要管道去連接即 FIFO 和 PIPO ,FIFO 是一個(gè)先進(jìn)先出存儲(chǔ)器也就是說使用這樣的管道傳輸數(shù)據(jù)的時(shí)候,數(shù)據(jù)進(jìn)出的順序不可以改變。而 PIPO 就是一個(gè)可尋址的存儲(chǔ)器管道,數(shù)據(jù)在任務(wù)之間進(jìn)出的順序可以改變。

最糟糕的狀態(tài)是什么?馬車在時(shí)間線上順序出發(fā),A 馬車到達(dá)終點(diǎn)后 B 再出發(fā)以此類推,就像是 CPU 中的單進(jìn)程順序執(zhí)行模式一樣,而FPGA中有可供并行化執(zhí)行的數(shù)據(jù)傳輸管道,更多的資源就像是跑道一樣,所以這個(gè)狀態(tài)效率是最低的。

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

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

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

奔跑的馬車帶著我們理解了任務(wù)級(jí)流水線的優(yōu)化之路,下面我們結(jié)合代碼看一看HLS工具會(huì)在哪些情況下阻止 dataflow 的實(shí)現(xiàn)。

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

以下是阻止任務(wù)級(jí)別并行性的常見情況:

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

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

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. 旁路任務(wù) Bypassing Tasks

正常情況下我們期望流水線任務(wù)是一個(gè)接著一個(gè)的產(chǎn)出并消耗,然而像下面這個(gè)例子中,Loop1 產(chǎn)生了 Temp1和Temp2 兩個(gè)數(shù)據(jù)流,但是在下一個(gè)任務(wù) Loop2 中只有 temp1 參與了運(yùn)算,而 temp2 就被旁支了。Loop3 任務(wù)的執(zhí)行依賴 Loop2 任務(wù)產(chǎn)生的 temp3 數(shù)據(jù),所以 Loop2 和 Loop3 因?yàn)閿?shù)據(jù)依賴的關(guān)系無法并行執(zhí)行。

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. 任務(wù)間雙向反饋 Feedbackbetween Tasks

假如說當(dāng)前任務(wù)的結(jié)果,需要作為之前一個(gè)任務(wù)的輸入的話,就形成了任務(wù)之間的數(shù)據(jù)反饋,它打亂了流水線從上級(jí)一直往下級(jí)輸送數(shù)據(jù)流的規(guī)則。這時(shí)候 HLS 就會(huì)給出警告或者報(bào)錯(cuò),有可能完成不了 dataflow 優(yōu)化了。有一種特例是支持的:使用 hls::stream 格式的數(shù)據(jù)流反饋。

我們分析以下代碼的內(nèi)容:

當(dāng)?shù)谝粋€(gè)程序 firstProc 執(zhí)行的時(shí)候,hls::stream 格式的數(shù)據(jù)流 forwardOUT 被寫入了初始化為10的數(shù)值 fromSecond 。由于 hls::stream 格式的數(shù)據(jù)本身不支持初始化操作,所以這樣的操作避免了違反單產(chǎn)出單消耗原則。之后的迭代里,firstProc 通過 backwardIN 接口從 hls :: stream 讀取數(shù)值寫入 forwardOUT 中。

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

#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. 含有條件判斷的任務(wù)流水

DATAFLOW 優(yōu)化不會(huì)優(yōu)化有條件執(zhí)行的任務(wù)。下面的示例展現(xiàn)了這個(gè)違例。在此示例中,有條件地執(zhí)行 Loop1 和 Loop2 會(huì)阻止 Vitis HLS 優(yōu)化這些循環(huán)之間的數(shù)據(jù)流,因?yàn)?sel 條件直接控制了任務(wù)中的數(shù)據(jù)有可能不會(huì)從一個(gè)循環(huán)流到下一個(gè)循環(huán)。

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]; } }

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

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. 有多種退出機(jī)制的循環(huán)

含有多種退出機(jī)制的循環(huán)不能被包含在流水線區(qū)域內(nèi),我們來數(shù)一數(shù) Loop2 一共有多少種循環(huán)退出條件:

1. 由 for 循環(huán)定義的 K>N 的情況;

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

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

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

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; } } }

我們理解了可能阻止任務(wù)流水線的 5 種經(jīng)典情況后,我們最后推出適用于 Vitis HLS 的Dataflow 優(yōu)化的兩種規(guī)范形式 (canonical forms) ,一種直接應(yīng)用于函數(shù),一種應(yīng)用于 for循環(huán)。我們可以發(fā)現(xiàn)規(guī)范形式嚴(yán)格遵守了單產(chǎn)出單消耗的規(guī)則。

1. 適用于子程序沒有被內(nèi)聯(lián) (inline) 的規(guī)范形式

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. 適用于循環(huán)體內(nèi)的任務(wù)流水的規(guī)范形式:

對于 for 循環(huán) (其中沒有內(nèi)聯(lián)函數(shù)的地方),循環(huán)變量應(yīng)具有:

a. 在 for 循環(huán)的標(biāo)題中聲明初始值,并設(shè)置為 0。

b. 循環(huán)條件N是一個(gè)正數(shù)值常數(shù)或常數(shù)函數(shù)參數(shù)。

c. 循環(huán)的遞增量為1。

d. Dataflow 指令必須位于循環(huán)內(nèi)部。

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); } }

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

審核編輯:何安

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

    關(guān)注

    0

    文章

    4

    瀏覽量

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

掃碼添加小助手

加入工程師交流群

    評論

    相關(guān)推薦
    熱點(diǎn)推薦

    步進(jìn)電機(jī)控制指令詳解

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

    NVME控制器設(shè)計(jì)之指令控制

    指令控制模塊由一個(gè)指令信息緩存, 一個(gè)指令組裝狀態(tài)機(jī)和一個(gè) ID 池組成。 指令信息緩存中存放著由系統(tǒng)控制模塊寫入的待處理指令信息;
    的頭像 發(fā)表于 04-24 10:22 ?280次閱讀
    NVME控制器設(shè)計(jì)之<b class='flag-5'>指令</b>控制

    CPU的各種指令和執(zhí)行流程

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

    驅(qū)動(dòng)板和屏幕的面板類型有關(guān)系嗎

    驅(qū)屏板和屏幕的面板類型有關(guān)系嗎?明思銳液晶屏方案商告訴你。沒有關(guān)系!驅(qū)屏板和屏幕的接口有關(guān),和面板類型無關(guān)。 ? 屏幕面板類型包括oled、IPS、VA和TN,但這些都和顯示屏驅(qū)動(dòng)板沒什么
    的頭像 發(fā)表于 01-07 14:45 ?841次閱讀

    RISC-V基礎(chǔ)指令詳解

    RISC-V中定義了六種指令類型,其中包括 R型指令 : 用于寄存器與寄存器之間算數(shù)運(yùn)算的指令 I型指令 : 用于寄存器和立即數(shù)算術(shù)運(yùn)算和讀存儲(chǔ)器操作的
    發(fā)表于 12-29 18:47

    Erp指令能效

    ErP指令EC244/2009、EC245/2009、EU1194/2012和能效標(biāo)簽指令EU874/2012已經(jīng)實(shí)行多年,歐盟委員會(huì)在近3年通過參照不斷改進(jìn)的照明產(chǎn)品技術(shù)、環(huán)境和經(jīng)濟(jì)因素以及實(shí)際
    的頭像 發(fā)表于 11-20 23:57 ?568次閱讀
    Erp<b class='flag-5'>指令</b>能效

    匯川INOVANCE 中型PLC指令手冊

    匯川INOVANCE 中型PLC指令手冊
    發(fā)表于 11-19 13:45 ?6次下載

    plc基本指令的應(yīng)用有哪些

    PLC(可編程邏輯控制器)的基本指令是PLC編程語言的基本元素,用于描述PLC如何根據(jù)輸入信號(hào)的狀態(tài)控制輸出信號(hào)的變化,從而實(shí)現(xiàn)對機(jī)械設(shè)備的控制。PLC的基本指令涵蓋了邏輯運(yùn)算、定時(shí)控制、計(jì)數(shù)
    的頭像 發(fā)表于 10-21 17:16 ?2761次閱讀

    處理器指令的獲取過程

    微處理器指令的獲取是計(jì)算機(jī)執(zhí)行程序過程中的關(guān)鍵環(huán)節(jié),它決定了微處理器如何對數(shù)據(jù)和指令進(jìn)行處理。以下將詳細(xì)闡述微處理器指令的獲取過程,包括指令的來源、存儲(chǔ)位置、讀取方式以及相關(guān)的硬件和軟
    的頭像 發(fā)表于 10-05 15:16 ?985次閱讀

    微處理器的指令集和指令系統(tǒng)有什么不同

    微處理器的指令集和指令系統(tǒng)是兩個(gè)緊密相關(guān)但又有所區(qū)別的概念,它們在微處理器的設(shè)計(jì)和運(yùn)行中扮演著不同的角色。以下是對這兩個(gè)概念的詳細(xì)解析,旨在深入探討它們之間的不同點(diǎn)。
    的頭像 發(fā)表于 10-05 14:57 ?1116次閱讀

    PLC指令的簡要說明

    PLC(可編程邏輯控制器)的指令雖然因不同的制造商和型號(hào)而有所差異,但有一些指令是相對比較通用的。
    的頭像 發(fā)表于 09-18 14:15 ?1466次閱讀

    復(fù)雜指令集和精簡指令集有什么區(qū)別

    復(fù)雜指令集(CISC,Complex Instruction Set Computer)和精簡指令集(RISC,Reduced Instruction Set Computer)是微處理器設(shè)計(jì)中
    的頭像 發(fā)表于 08-22 11:00 ?5723次閱讀

    西門子移位指令怎么用在順序控制

    西門子的PLC(可編程邏輯控制器)廣泛應(yīng)用于工業(yè)自動(dòng)化領(lǐng)域,其指令集豐富,能夠滿足各種控制需求。在順序控制中,移位指令是一種非常有用的指令,可以實(shí)現(xiàn)數(shù)據(jù)的移動(dòng)和處理。 移位指令概述 移
    的頭像 發(fā)表于 08-16 17:08 ?2084次閱讀

    三菱 FX 系列PLC的基本邏輯指令(2)

    ?置位與復(fù)位指令(SET/RST) ?(1)SET(置位指令) 它的作用是使被操作的目標(biāo)元件置位并保持。 ?(2)RST(復(fù)位指令) 使被操作的目標(biāo)元件復(fù)位并保持清零狀態(tài)。SET、RST指令
    的頭像 發(fā)表于 08-05 09:06 ?1373次閱讀

    渦流損耗的大小與什么有關(guān)

    渦流損耗,又稱為渦流損耗或渦流熱損失,是指在導(dǎo)體中由于交變磁場的作用而產(chǎn)生的渦流所引起的能量損耗。渦流損耗在許多領(lǐng)域都有應(yīng)用,如電機(jī)、變壓器、感應(yīng)加熱等。本文將探討渦流損耗的大小與哪些因素有關(guān)
    的頭像 發(fā)表于 07-26 16:43 ?3802次閱讀