《電子技術應用》
您所在的位置:首頁 > 可編程邏輯 > 解決方案 > 使用Vivado HLS實現OpenCV的開發流程

使用Vivado HLS實現OpenCV的開發流程

2014-01-02
作者:Harvest Guo
來源:Xilinx DSP Specilist
關鍵詞: 開發工具 Vivado HLS opencv

    本文通過對OpenCV中圖像類型和函數處理方法的介紹,通過設計實例描述在vivadoHLS中調用OpenCV庫函數實現圖像處理的幾個基本步驟,完成從OpenCV設計到RTL轉換綜合的開發流程。

    開源計算機視覺 (OpenCV) 被廣泛用于開發計算機視覺應用,它包含2500多個優化的視頻函數的函數庫并且專門針對臺式機處理器和GPU進行優化。OpenCV的用戶成千上萬,OpenCV的設計無需修改即可在 Zynq器件的ARM處理器上運行。但是利用OpenCV實現的高清處理經常受外部存儲器的限制,尤其是存儲帶寬會成為性能瓶頸,存儲訪問也會限制功耗效率。使用VivadoHLS高級語言綜合工具,可以輕松實現OpenCV C++視頻處理設計到RTL代碼的轉換,輸出硬件加速器或者直接在FPGA上實現實時視頻處理功能。同時,Zynq All-programmable SOC是實現嵌入式計算機視覺應用的極好方法,很好解決了在單一處理器上實現視頻處理性能低功耗高的限制,Zynq高性能可編程邏輯和嵌入式ARM內核,是一款功耗優化的集成式解決方案。

1   OpenCV中圖像IplImage, CvMat, Mat 類型的關系和VivadoHLS中圖像hls::Mat類型介紹

    OpenCv中常見的與圖像操作有關的數據容器有Mat,cvMat和IplImage,這三種類型都可以代表和顯示圖像,但是,Mat類型側重于計算,數學性較高,openCV對Mat類型的計算也進行了優化。而CvMat和IplImage類型更側重于“圖像”,opencv對其中的圖像操作(縮放、單通道提取、圖像閾值操作等)進行了優化。在opencv2.0之前,opencv是完全用C實現的,但是,IplImage類型與CvMat類型的關系類似于面向對象中的繼承關系。實際上,CvMat之上還有一個更抽象的基類----CvArr,這在源代碼中會常見。

1.1 OpenCV中Mat類型:矩陣類型(Matrix)。

    在openCV中,Mat是一個多維的密集數據數組。可以用來處理向量和矩陣、圖像、直方圖等等常見的多維數據。

    Mat有3個重要的方法:

   1、Mat mat = imread(const String* filename);            讀取圖像

  2、imshow(const string frameName, InputArray mat);     顯示圖像

  3、imwrite (const string& filename, InputArray img);    儲存圖像

    Mat類型較CvMat與IplImage類型來說,有更強的矩陣運算能力,支持常見的矩陣運算。在計算密集型的應用當中,將CvMat與IplImage類型轉化為Mat類型將大大減少計算時間花費。

1.2 OpenCV中CvMat類型與IplImage類型:“圖像”類型

    在openCV中,Mat類型與CvMat和IplImage類型都可以代表和顯示圖像,但是,Mat類型側重于計算,數學性較高,openCV對Mat類型的計算也進行了優化。而CvMat和IplImage類型更側重于“圖像”,openCV對其中的圖像操作(縮放、單通道提取、圖像閾值操作等)進行了優化。

補充:IplImage由CvMat派生,而CvMat由CvArr派生即CvArr -> CvMat -> IplImage

     CvArr用作函數的參數,無論傳入的是CvMat或IplImage,內部都是按CvMat處理。

    在openCV中,沒有向量(vector)的數據結構。任何時候,但我們要表示向量時,用矩陣數據表示即可。

    但是,CvMat類型與我們在線性代數課程上學的向量概念相比,更抽象,比如CvMat的元素數據類型并不僅限于基礎數據類型,比如,下面創建一個二維數據矩陣:

      CvMat* cvCreatMat(int rows ,int cols , int type);

這里的type可以是任意的預定義數據類型,比如RGB或者別的多通道數據。這樣我們便可以在一個CvMat矩陣上表示豐富多彩的圖像了。

1.3 OpenCV中IplImage類型

    在OpenCV類型關系上,我們可以說IplImage類型繼承自CvMat類型,當然還包括其他的變量將之解析成圖像數據。

    IplImage類型較之CvMat多了很多參數,比如depth和nChannels。在普通的矩陣類型當中,通常深度和通道數被同時表示,如用32位表示RGB+Alpha.但是,在圖像處理中,我們往往將深度與通道數分開處理,這樣做是OpenCV對圖像表示的一種優化方案。

IplImage的對圖像的另一種優化是變量origin----原點。在計算機視覺處理上,一個重要的不便是對原點的定義不清楚,圖像來源,編碼格式,甚至操作系統都會對原地的選取產生影響。為了彌補這一點,openCV允許用戶定義自己的原點設置。取值0表示原點位于圖片左上角,1表示左下角。

1.4 VivadoHLS中圖像數據類型hls::Mat<>

    VivadoHLS視頻處理函數庫使用hls::Mat<>數據類型,這種類型用于模型化視頻像素流處理,實質等同于hls::steam<>流的類型,而不是OpenCV中在外部memory中存儲的matrix矩陣類型。因此,在HLS實現OpenCV的設計中,需要將輸入和輸出HLS可綜合的視頻設計接口,修改為Video stream接口,也就是采用HLS提供的video接口可綜合函數,實現AXI4 video stream到VivadoHLS中hls::Mat<>類型的轉換。

2   使用VivadoHLS實現OpenCV到RTL代碼轉換的流程

2.1 OpenCV設計中的權衡

    OpenCV圖像處理是基于存儲器幀緩存而構建的,它總是假設視頻frame數據存放在外部DDR 存儲器中,因此,OpenCV對于訪問局部圖像性能較差,因為處理器的小容量高速緩存性能不足以完成這個任務。而且出于性能考慮,基于OpenCV設計的架構比較復雜,功耗更高。在對分辨率或幀速率要求低,或者在更大的圖像中對需要的特征或區域進行處理是,OpenCV似乎足以滿足很多應用的要求,但對于高分辨率高幀率實時處理的場景下,OpenCV很難滿足高性能和低功耗的需求。

基于視頻流的架構能提供高性能和低功耗,鏈條化的圖像處理函數能減少外部存儲器訪問,針對視頻優化的行緩存和窗口緩存比處理器高速緩存更簡單,更易于用FPGA部件,使用VivadoHLS中的數據流優化來實現.

VivadoHLS對OpenCV的支持,不是指可以將OpenCV的函數庫直接綜合成RTL代碼,而是需要將代碼轉換為可綜合的代碼,這些可綜合的視頻庫稱為HLS視頻庫,由VivadoHLS提供。

    OpenCV函數不能直接通過HLS進行綜合,因為OpenCV函數一般都包含動態的內存分配、浮點以及假設圖像在外部存儲器中存放或者修改。

    VivadoHLS視頻庫用于替換很多基本的 OpenCV函數,它與OpenCV具有相似的接口和算法,主要針對在FPGA架構中實現的圖像處理函數,包含了專門面向FPGA的優化,比如定點運算而非浮點運算(不必精確到比特位),片上的行緩存(line buffer)和窗口緩存(window buffer)。

2.2 VivadoHLS實現OpenCV設計流程介紹

使用VivadoHLS實現OpenCV的開發,主要的三個步驟如下:

  1. 在計算機上開發OpenCV應用,由于是開源的設計,采用C++的編譯器對其進行編譯,仿真和debug,最后產生可執行文件。這些設計無需修改即可在 ARM內核上運行OpenCV應用。
  2. 使用I/O函數抽取FPGA實現的部分,并且使用可綜合的VivadoHLS Video庫函數代碼代替OpenCV函數的調用。
  3. 運行HLS生成RTL代碼,在vivadoHLS工程中啟動co-sim,重用openCV的測試激勵驗證產生的RTL代碼。在ISE或者Vivado開發環境中做RTL的集成和SOC/FPGA實現。

2.2.1        VivadoHLS視頻庫函數

HLS視頻庫是包含在hls命名空間內的C++代碼。#include “hls_video.h”

與OpenCV等具有相似的接口和等效的行為,例如:

           OpenCV庫:cvScale(src, dst, scale, shift);

           HLS視頻庫:hls::catfrustrated:cale<...>(src, dst, scale, shift);

一些構造函數具有類似的或替代性的模板參數,例如:

           OpenCV庫:cv::Mat mat(rows, cols, CV_8UC3);

           HLS視頻庫:hls::Mat mat(rows, cols);

                      ROWS和COLS指定處理的最大圖像尺寸

 

              表2.2.1 VivadoHLS視頻處理函數庫

 

2.2.2        VivadHLS實現OpenCV設計的局限性

    首先,必須用HLS視頻庫函數代替OpenCV調用。

    其次,不支持OpenCV通過指針訪問幀緩存,可以在HLS中使用VDMA和 AXI Stream adpater函數代替。

    再者,不支持OpenCV的隨機訪問。HLS對于讀取超過一次的數據必須進行復制,更多的例子可以參見見hls::catvery-happy:uplicate()函數。

最后,不支持OpenCVS的In-place更新,比如  cvRectangle (img, point1, point2)。

 

下面表格2.2.2列舉了OpenCV中隨機訪問一幀圖像處理對應HLS視頻庫的實現方法。

 

 

OpenCV

HLS視頻庫

讀操作

pix = cv_mat.at(i,j)

pix = cvGet2D(cv_img,i,j)

hls_img >> pix

寫操作

cv_mat.at(i,j) = pix

cvSet2D(cv_img,i,j,pix)

hls_img << pix

               表 2.2.2 OpenCV和HLS中對一幀圖像像素訪問對應方法

2.3 用HLS實現OpenCV應用的實例(快速角點濾波器image_filter)

    我們通過快速角點的例子,說明通常用VivadoHLS實現OpenCV的流程。首先,開發基于OpenCV的快速角點算法設計,并使用基于OpenCV的測試激勵仿真驗證這個算法。接著,建立基于視頻數據流鏈的OpenCV處理算法,改寫前面直覺的OpenCV的通常設計,這樣的改寫是為了與HLS視頻庫處理機制相同,方便后面步驟的函數替換。最后,將改寫的OpenCV設計中的函數,替換為HLS提供的相應功能的視頻函數,并用VivadoHLS綜合,最后在Xilinx開發環境下實現。當然,這些可綜合代碼也可在處理器或ARM上運行。

2.3.1        設計基于OpenCV的視頻濾波器設計和測試激勵

    在這個例子中,首先設計開發完全調用OpenCV庫函數的快速角點濾波器設計opencv_image_filter.cpp和這個濾波器的測試激勵opencv_image_filter_tb.cpp,測試激勵用于仿真驗證opencv_image_filter算法功能。算法和測試激勵設計代碼如下:

 

void opencv_image_filter(IplImage* src, IplImage* dst)

{

    IplImage* gray = cvCreateImage( cvGetSize(src), 8, 1 ); 

    std::vector keypoints;

cv::Mat gray_mat(gray,0);

 

    cvCvtColor( src, gray, CV_BGR2GRAY ); 

    cv::FAST( gray_mat, keypoints, 20, true);

    cvCopy( src,dst);

 

    for (int i=0;i

    {

      cvRectangle(dst, cvPoint(keypoints[i].pt.x-1,keypoints[i].pt.y-1),

                                cvPoint(keypoints[i].pt.x+1,keypoints[i].pt.y+1), cvScalar(255,0,0),CV_FILLED);

    }

    cvReleaseImage( &gray ); 

}   

       例子2.3.1.1 通常的OpenCV視頻處理代碼opencv_image_filter.cpp

 

int main (int argc, char** argv) {

    IplImage* src=cvLoadImage(INPUT_IMAGE);

    IplImage* dst = cvCreateImage(cvGetSize(src), src->depth, src->nChannels);

        

    opencv_image_filter(src, dst);

    cvSaveImage(OUTPUT_IMAGE_GOLDEN, dst);

 

    cvReleaseImage(&src);

    cvReleaseImage(&dst);

    return 0;

}

   例子2.3.1.2 OpenCV視頻處理測試激勵代碼opencv_image_filter_tb.cpp

 

    上面的例子是直接調用OpenCV在處理器上軟件應用實現的例子,可以看到在算法設計中直接調用opencV庫函數,測試激勵讀入圖像,經過濾波器處理輸出的圖像保存分析??梢钥吹?,算法的處理基于IPIimage類型,輸入和輸出圖像都使用此類型。

 

2.3.2        使用IO函數和Vivado HLS視頻庫替換OpenCV函數庫

    需要特別說明的是,xilinx通常使用的視頻處理模塊都是基于axi4 streaming協議進行不同模式見像素數據的交互,也就是我們所說的AXI4 video接口協議格式。為了和xilinx視頻庫接口協議統一,VivadoHLS提供了視頻接口函數庫,用于從OpenCV程序中抽取需要進行RTL綜合轉換的頂層函數,并把這些可綜合的代碼和OpenCV不可綜合轉換的代碼進行隔離。然后,對需要綜合轉換為RTL代碼的OpenCV函數,用xilinx VivadoHLS提供相應功能的可綜合video函數進行替換。最后在C/C++編譯環境下仿真驗證OpenCV代碼和替換video函數后功能的一致,并在VivadoHLS開發環境中做代碼綜合和產生RTL代碼的co-sim混合仿真驗證。

 

VivadoHLS可綜合的視頻接口函數:

Hls::AXIvideo2Mat  轉換AXI4 video stream到hls::Mat表示格式

Hls::Mat2AXIvideo  轉換hls::Mat數據格式到AXI4 video stream

 

    首先,我們對2.3.1中OpenCV的設計進行改寫,改寫的代碼還是完全基于OpenCV的函數,目的是為了對視頻的處理機制基于視頻流的方式,與VivadoHLS視頻庫提供函數的處理機制一致。下面是OpenCV設計的另一種寫法:

 

void opencv_image_filter(IplImage* src, IplImage* dst)

{

    IplImage* gray = cvCreateImage( cvGetSize(src), 8, 1 ); 

    IplImage* mask = cvCreateImage( cvGetSize(src), 8, 1 ); 

    IplImage* dmask = cvCreateImage( cvGetSize(src), 8, 1 ); 

    std::vector keypoints;

    cv::Mat gray_mat(gray,0);

   

    cvCvtColor(src, gray, CV_BGR2GRAY ); 

    cv::FAST(gray_mat, keypoints, 20, true);

    GenMask(mask, keypoints);

    cvDilate(mask,dmask);

    cvCopy(src,dst);

    PrintMask(dst,dmask,cvScalar(255,0,0));

 

    cvReleaseImage( &mask ); 

    cvReleaseImage( &dmask ); 

    cvReleaseImage( &gray ); 

}

例子2.3.2.1另一種OpenCV設計應用opencv_image_filter.cpp

 

    其次,使用Vivado HLS視頻庫替代標準OpenCV函數,并使用可綜合的視頻接口函數,采用video stream的方式交互視頻數據。用于FPGA的硬件可綜合模塊由VivadoHLS視頻庫函數與接口組成,我們用hls命名空間中的相似函數代替OpenCV函數,增加接口函數構建AXI4 stream類型的接口。

 

void image_filter(AXI_STREAM& input, AXI_STREAM& output, int rows, int cols)

{

//Create AXI streaming interfaces for the core

#pragma HLS RESOURCE variable=input core=AXIS metadata="-bus_bundle INPUT_STREAM"

#pragma HLS RESOURCE variable=output core=AXIS metadata="-bus_bundle OUTPUT_STREAM"

#pragma HLS RESOURCE core=AXI_SLAVE variable=rows metadata="-bus_bundle CONTROL_BUS"

#pragma HLS RESOURCE core=AXI_SLAVE variable=cols metadata="-bus_bundle CONTROL_BUS"

#pragma HLS RESOURCE core=AXI_SLAVE variable=return metadata="-bus_bundle CONTROL_BUS"

 

#pragma HLS interface ap_stable port=rows

#pragma HLS interface ap_stable port=cols

    hls::Mat      _src(rows,cols);

    hls::Mat      _dst(rows,cols);

#pragma HLS dataflow

    hls::AXIvideo2Mat(input, _src);

    hls::Mat      src0(rows,cols);

    hls::Mat      src1(rows,cols);

#pragma HLS stream depth=20000 variable=src1.data_stream

    hls::Mat      mask(rows,cols);

    hls::Mat      dmask(rows,cols);

    hls::catfrustrated:calar<3,unsigned char> color(255,0,0);

    hls::catvery-happy:uplicate(_src,src0,src1);

    hls::Mat      gray(rows,cols);

    hls::CvtColor(src0,gray);

    hls::FASTX(gray,mask,20,true);

    hls::catvery-happy:ilate(mask,dmask);

    hls::cattongue:aintMask(src1,dmask,_dst,color);

    hls::Mat2AXIvideo(_dst, output);

}

例子2.3.2.2 采用VivadoHLS視頻庫替換后可綜合的設計opencv_image_filter.cpp

 

  最后,在vivadoHLS開發環境下綜合例子2.3.2.2的設計,產生RTL代碼并重用OpenCV的測試激勵驗證RTL代碼功能。

3   VHLS實現OpenCV設計流程總結

    通過上面章節介紹以及在vivadoHLS工具中實現opencV設計的例子可以看出,OpenCV函數可實現計算機視覺算法的快速原型設計,并使用VivadoHLS工具轉換為RTL代碼在FPGA或者Zynq SOC上實現高分辨率高幀率的實時視頻處理。計算機視覺應用與生俱來的異構特性,使其需要軟硬件相結合的實現方案。Vivado HLS視頻庫能加快OpenCV函數向FPGA可編程架構的映射。

 

 

本站內容除特別聲明的原創文章之外,轉載內容只為傳遞更多信息,并不代表本網站贊同其觀點。轉載的所有的文章、圖片、音/視頻文件等資料的版權歸版權所有權人所有。本站采用的非本站原創文章及圖片等內容無法一一聯系確認版權者。如涉及作品內容、版權和其它問題,請及時通過電子郵件或電話通知我們,以便迅速采取適當措施,避免給雙方造成不必要的經濟損失。聯系電話:010-82306118;郵箱:aet@chinaaet.com。
亚洲一区二区欧美_亚洲丝袜一区_99re亚洲国产精品_日韩亚洲一区二区
国产视频精品xxxx| 亚洲激情在线播放| 欧美精品色一区二区三区| 久久久久成人精品| 久久精品观看| 久久精品国产第一区二区三区最新章节 | 亚洲在线国产日韩欧美| 日韩视频中文字幕| 日韩一级黄色片| 99re6这里只有精品| 日韩视频在线免费观看| 日韩西西人体444www| 日韩一区二区精品视频| 99re6热在线精品视频播放速度| 亚洲精品一区二| 亚洲最新在线视频| 亚洲视频久久| 午夜精品亚洲| 欧美中文字幕在线观看| 亚洲国产电影| 日韩视频一区| 在线天堂一区av电影| 亚洲一二三区在线观看| 午夜视频一区在线观看| 久久av资源网| 美女露胸一区二区三区| 欧美精品色综合| 国产精品久久| 国产一区二区精品| 亚洲国产欧美日韩| 亚洲免费精品| 亚洲欧美日本在线| 亚洲福利专区| a91a精品视频在线观看| 亚洲欧美综合v| 久久婷婷人人澡人人喊人人爽| 免费在线国产精品| 欧美三级午夜理伦三级中文幕| 国产精品大片| 国产一区清纯| 亚洲精品免费在线播放| 亚洲专区欧美专区| 最新成人在线| 亚洲男人的天堂在线观看| 久久精品成人一区二区三区 | 狠狠色2019综合网| 亚洲精品视频免费观看| 亚洲自拍偷拍福利| 亚洲黄色成人久久久| 亚洲一区二区精品在线观看| 久久精品夜夜夜夜久久| 欧美裸体一区二区三区| 国产美女精品视频| 亚洲国产成人精品久久久国产成人一区| 亚洲精品日韩一| 性欧美超级视频| 日韩视频在线免费观看| 午夜精品久久| 欧美风情在线观看| 国产精品永久| 亚洲精品欧美日韩| 欧美专区在线| 亚洲一区在线播放| 蜜桃av一区二区在线观看| 国产精品成人va在线观看| 亚洲第一黄色网| 亚洲一区二区视频| 亚洲精品永久免费| 久久久久在线观看| 国产精品啊v在线| 亚洲国产精品一区二区www在线 | 午夜久久影院| 在线一区视频| 免费成人在线观看视频| 国产欧美精品一区二区三区介绍| 亚洲国产欧美国产综合一区| 午夜激情亚洲| 在线综合视频| 免费欧美日韩国产三级电影| 国产区日韩欧美| 夜夜夜久久久| 亚洲日本欧美日韩高观看| 久久国产精品一区二区| 欧美三区在线| 亚洲国产欧美一区| 久久国内精品视频| 午夜亚洲性色视频| 欧美午夜精品理论片a级大开眼界| 影音先锋一区| 久久精品国产综合| 欧美在线一二三| 国产精品乱人伦一区二区| 亚洲麻豆一区| 日韩一级黄色av| 欧美电影免费观看网站| 国产自产精品| 午夜精品一区二区三区电影天堂| 一区二区三区日韩欧美精品| 欧美成人精品三级在线观看| 狠狠色噜噜狠狠色综合久| 香蕉免费一区二区三区在线观看| 亚洲一本大道在线| 欧美日韩另类一区| 亚洲精品一区二区网址| 日韩特黄影片| 欧美激情综合五月色丁香| 亚洲电影一级黄| 亚洲精品日韩综合观看成人91| 久久人人九九| 国产在线精品自拍| 欧美一区二区高清在线观看| 亚洲欧美日韩久久精品| 欧美午夜视频网站| 日韩亚洲国产欧美| 一区二区三区国产盗摄| 欧美久色视频| 亚洲三级视频| 亚洲视频免费看| 欧美色视频在线| 一区二区精品国产| 亚洲一区激情| 国产精品日韩电影| 香蕉精品999视频一区二区| 欧美一区二区视频97| 国产日韩视频| 久久国产免费看| 久久只精品国产| 在线播放国产一区中文字幕剧情欧美| 久久精品亚洲国产奇米99| 玖玖视频精品| 亚洲激情视频在线| 一区二区三区波多野结衣在线观看| 欧美日本精品在线| 一本到12不卡视频在线dvd| 亚洲一区二区毛片| 国产精品三上| 欧美一区二视频在线免费观看| 久久久九九九九| 亚洲国产精品久久久久久女王| 一区二区三区国产在线| 国产精品毛片| 久久本道综合色狠狠五月| 免费一级欧美片在线观看| 91久久久在线| 亚洲中字在线| 韩国免费一区| 99riav久久精品riav| 国产精品乱人伦一区二区| 欧美一区二区三区久久精品| 欧美va天堂| 99在线视频精品| 欧美一区二区在线免费观看 | 亚洲一区二区三区在线看| 久久精品国产精品亚洲综合| 一区二区三区在线不卡| 日韩一级视频免费观看在线| 国产精品国产三级欧美二区| 亚洲欧美日韩天堂一区二区| 免费视频一区二区三区在线观看| 亚洲精品资源| 欧美在线中文字幕| 亚洲国产欧美在线| 午夜欧美精品久久久久久久| 好看的亚洲午夜视频在线| 一区二区三区日韩精品视频| 国产精品一区在线观看| 亚洲精品1区2区| 国产精品久久一级| 久久国产精品久久久久久电车| 欧美美女视频| 久久成人精品无人区| 欧美视频免费看| 欧美在线视频一区二区| 欧美日韩国产电影| 久久成人18免费观看| 欧美视频在线观看| 久久精品噜噜噜成人av农村| 欧美日本不卡高清| 久久精品国产一区二区电影| 欧美日韩亚洲综合一区| 久久不见久久见免费视频1| 欧美日韩日日夜夜| 久久精品午夜| 国产精品视频xxx| 亚洲精品国产系列| 国产日本欧美一区二区三区| 日韩一级黄色片| 韩国美女久久| 亚洲欧美一区二区在线观看| 亚洲国产精品久久人人爱蜜臀| 欧美一区国产二区| 99视频在线观看一区三区| 久久视频国产精品免费视频在线| 日韩一区二区免费高清| 麻豆精品在线视频| 欧美亚洲在线观看| 国产精品chinese| 夜夜嗨av一区二区三区中文字幕| 极品av少妇一区二区| 久久精品国产一区二区三区免费看 |