本博文採用Xilinx HLS 2014.4工具。实现一个肤色检測的模块。当中,本文重点是构建HLS图像处理函数。

新建HLSproject的步骤,本博文不再详述。

本project新建之后,仅仅加入了五个文件,例如以下图所看到的。当中,top.cpp中的主函数终于会综合生成HLS硬件图像处理模块。test.cpp是測试文件,调用測试图片。測试top.cpp的图像处理函数功能。

top.cpp的源代码例如以下:

#include "top.h"
#include "imgprocess.h"
#include <string.h> void ImgProcess_Top(AXI_STREAM& input, AXI_STREAM& output,int rows, int cols,
int y_lower,int y_upper,int cb_lower,int cb_upper,int cr_lower,int cr_upper)
{
#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=y_lower metadata="-bus_bundle CONTROL_BUS"
#pragma HLS RESOURCE core=AXI_SLAVE variable=y_upper metadata="-bus_bundle CONTROL_BUS"
#pragma HLS RESOURCE core=AXI_SLAVE variable=cb_lower metadata="-bus_bundle CONTROL_BUS"
#pragma HLS RESOURCE core=AXI_SLAVE variable=cb_upper metadata="-bus_bundle CONTROL_BUS"
#pragma HLS RESOURCE core=AXI_SLAVE variable=cr_lower metadata="-bus_bundle CONTROL_BUS"
#pragma HLS RESOURCE core=AXI_SLAVE variable=cr_upper 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
#pragma HLS INTERFACE ap_stable port=y_lower
#pragma HLS INTERFACE ap_stable port=y_upper
#pragma HLS INTERFACE ap_stable port=cb_lower
#pragma HLS INTERFACE ap_stable port=cb_upper
#pragma HLS INTERFACE ap_stable port=cr_lower
#pragma HLS INTERFACE ap_stable port=cr_upper
RGB_IMAGE src_mat(rows,cols);
RGB_IMAGE dst_mat(rows,cols);
#pragma HLS dataflow
hls::AXIvideo2Mat(input, src_mat);
SkinColorDetect(src_mat,dst_mat, y_lower, y_upper, cb_lower, cb_upper, cr_lower, cr_upper);
hls::Mat2AXIvideo(dst_mat, output);
}

当中。ImgProcess_Top这个函数最后生成一个IP核,能够放在图像通路中使用。函数的接口例如以下:

input:视频流输入,axi-stream接口;

output:视频流输出,axi-stream接口;

rows,cols:可配置參数,图像的行数、列数。

通过AXI-Lite接口,由PS配置。

y_lower,y_upper,cb_lower,cb_upper,cr_lower,cr_upper:可配置參数,肤色检測的一些阈值。通过AXI-Lite接口。由PS配置。

上述代码中,比較重要的一条优化指令为:#pragma HLS dataflow。

它使得任务之间为流水线方式,也就是hls::AXIvideo2Mat(input, src_mat);SkinColorDetect(src_mat,dst_mat, y_lower, y_upper, cb_lower, cb_upper, cr_lower, cr_upper);hls::Mat2AXIvideo(dst_mat, output);这三个函数之间为流水线方式运行。

肤色检測的核心函数为SkinColorDetect(src_mat,dst_mat, y_lower, y_upper, cb_lower, cb_upper, cr_lower, cr_upper);它包括在imgprocess.h源代码例如以下:

#ifndef ___IMAGEPROCESS__
#define ___IMAGEPROCESS__
#include "top.h" u1 rgb2ycbcr(u8 B, u8 G, u8 R, int y_lower, int y_upper, int cb_lower, int cb_upper, int cr_lower, int cr_upper)
{
#pragma HLS PIPELINE
u8 y, cr, cb;
y = (76 * R.to_int() + 150 * G.to_int() + 29 * B.to_int()) >> 8;
cb = 128 + ((128*B.to_int() -43*R.to_int() - 85*G.to_int())>>8);
cr = 128 + ((128*R.to_int() -107*G.to_int() - 21 * B.to_int())>>8); if (y > y_lower && y < y_upper && cb > cb_lower && cb < cb_upper
&& cr > cr_lower && cr < cr_upper)
return 1;
else
return 0;
} namespace hls {
template<int SRC_T, int DST_T,int ROW, int COL>
void ImgProcess(Mat<ROW, COL, SRC_T> &_src, Mat<ROW, COL, DST_T> &_dst,
int y_lower,int y_upper,int cb_lower,int cb_upper,int cr_lower,int cr_upper)
{
loop_height: for(HLS_SIZE_T i= 0;i< ROW;i++)
{
#pragma HLS LOOP_TRIPCOUNT max=ROW
loop_width: for (HLS_SIZE_T j= 0;j< COL;j++)
{
#pragma HLS LOOP_FLATTEN OFF
#pragma HLS LOOP_TRIPCOUNT max=COL
#pragma HLS DEPENDENCE array inter false
#pragma HLS PIPELINE
u8 r, g, b;
u1 skin_region; HLS_TNAME(SRC_T) temp0=0;
HLS_TNAME(SRC_T) temp1=0;
HLS_TNAME(SRC_T) temp2=0;
/***********stream input *********/
_src.data_stream[0]>>temp0;
_src.data_stream[1]>>temp1;
_src.data_stream[2]>>temp2; b = temp0;
g = temp1;
r = temp2;
/********detect skin region*******/
skin_region = rgb2ycbcr(b, g, r,y_lower,y_upper,cb_lower,cb_upper,cr_lower,cr_upper);
HLS_TNAME(DST_T) temp_dst0=0;
HLS_TNAME(DST_T) temp_dst1=0;
HLS_TNAME(DST_T) temp_dst2=0; temp_dst0= (skin_region == 1)? b : (u8)0;
temp_dst1= (skin_region == 1)? g : (u8)0;
temp_dst2= (skin_region == 1)? r : (u8)0; /***********stream output ********/
_dst.data_stream[0]<<temp_dst0;
_dst.data_stream[1]<<temp_dst1;
_dst.data_stream[2]<<temp_dst2; }
}
} template<int SRC_T, int DST_T,int ROW, int COL>
void SkinColorDetect(Mat<ROW, COL, SRC_T> &_src,Mat<ROW, COL, DST_T> &_dst,
int y_lower,int y_upper,int cb_lower,int cb_upper,int cr_lower,int cr_upper)
{
#pragma HLS INLINE
ImgProcess(_src, _dst, y_lower, y_upper, cb_lower, cb_upper, cr_lower, cr_upper);
} } #endif

核心函数是rgb2ycbcr这个函数。关于肤色检測有多种方式,本文的肤色检測方法是将rgb转换为ycbcr,然后设置阈值。

保存后,综合。

综合完成,打开分析工具:

点击红框里的内容:

能够看到imgprocess.h中,ImgProcess这个函数的运行状态:

然后点击ImgProcess_Top_rgb2ycbcr,能够看到例如以下图:

我们发现,仅仅需一个时钟周期就可以运行完成。这是由于rgb2ycbcr这个函数採用了流水线的优化指令:#pragma HLS PIPELINE。

综合之后,就能够測试了。

test.cpp内容例如以下:

#include "top.h"
#include "hls_opencv.h"
#include"iostream"
#include<time.h>
using namespace std;
using namespace cv; int main (int argc, char** argv) { Mat src = imread("test.jpg");
AXI_STREAM src_axi, dst_axi;
Mat dst(Size(640,480),CV_8UC3); resize(src,src,Size(640,480));
//mat to axi video
cvMat2AXIvideo(src, src_axi);
//test function
ImgProcess_Top(src_axi, dst_axi, 480, 640,0,255,80,135,131,185);
//axi video to mat
AXIvideo2cvMat(dst_axi, dst); imshow("src",src);
imshow("dst_hls",dst); waitKey(0); return 0;
}

測试的图像例如以下:

执行測试程序后。输出图像例如以下:

通过測试后,点击hls界面工具栏的export RTLbutton,打包生成ip。最后的IP例如以下所看到的:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvTFpZMjcyOTQyNTE4/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

HLS图像处理系列——肤色检測的更多相关文章

  1. 图像处理之Canny边缘检測

    图像处理之Canny 边缘检測 一:历史 Canny边缘检測算法是1986年有John F. Canny开发出来一种基于图像梯度计算的边缘 检測算法,同一时候Canny本人对计算图像边缘提取学科的发展 ...

  2. 爱国者布局智能硬件,空探系列PM2.5检測仪“嗅霾狗”大曝光

        随着6月1日史上最严禁烟令的正式实施,国内包含北京.上海.成都等大中型城市已经在公共场所全面禁烟.众所周知,实施禁烟令的根本在于促进空气的净化,实现环境的改善,要达到这个目的,光有禁烟令是远远 ...

  3. Python图像处理(8):边缘检測

    快乐虾 http://blog.csdn.net/lights_joy/ 欢迎转载,但请保留作者信息 此前已经得到了单个区域植株图像,接下来似乎应该尝试对这些区域进行分类识别.通过外形和叶脉进行植物种 ...

  4. OpenCV图像处理篇之边缘检測算子

    3种边缘检測算子 灰度或结构等信息的突变位置是图像的边缘,图像的边缘有幅度和方向属性.沿边缘方向像素变化缓慢,垂直边缘方向像素变化剧烈.因此,边缘上的变化能通过梯度计算出来. 一阶导数的梯度算子 对于 ...

  5. 【OpenCV新手教程之十二】OpenCV边缘检測:Canny算子,Sobel算子,Laplace算子,Scharr滤波器合辑

    本系列文章由@浅墨_毛星云 出品,转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/25560901 作者:毛星云(浅墨) ...

  6. Matlab图像处理系列2———空间域平滑滤波器

    注:本系列来自于图像处理课程实验,用Matlab实现最主要的图像处理算法 本文章是Matlab图像处理系列的第二篇文章.介绍了空间域图像处理最主要的概念----模版和滤波器,给出了均值滤波起和中值滤波 ...

  7. 【OpenCV新手教程之十七】OpenCV重映射 &amp; SURF特征点检測合辑

    本系列文章由@浅墨_毛星云 出品.转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/30974513 作者:毛星云(浅墨)  ...

  8. 目标检測的图像特征提取之(一)HOG特征

    1.HOG特征: 方向梯度直方图(Histogram of Oriented Gradient, HOG)特征是一种在计算机视觉和图像处理中用来进行物体检測的特征描写叙述子.它通过计算和统计图像局部区 ...

  9. AIX下RAC搭建 Oracle10G(一)检測系统环境

    AIX下RAC搭建系列 环境 节点 节点1 节点2 小机型号 IBM P-series 630 IBM P-series 630 主机名 AIX203 AIX204 交换机 SAN光纤交换机 存储 S ...

随机推荐

  1. Freemarker-2.3.22 Demo - No01_获取模板并直接输出

    package No01_获取模板并直接输出; import java.io.File; import java.io.FileOutputStream; import java.io.OutputS ...

  2. How to add elements to a List in Scala

    Scala List FAQ: How do I add elements to a Scala List? This is actually a trick question, because yo ...

  3. Ubuntu 12.04下mysql的安装与配置

    转自:http://blog.csdn.net/ichsonx/article/details/9285935 准备  0. 获取 mysql-5.5.15-linux2.6-i686.tar.gz ...

  4. GNU风格 ARM汇编语法4

    .GNU汇编语言定义入口点 汇编程序的缺省入口是_start标号,用户也可以在连接脚本文件中用ENTRY标志指明其它入口点. 例:定义入口点 .section .data < initializ ...

  5. Cannon

    Description In Chinese Chess, there is one kind of powerful chessmen called Cannon. It can move hori ...

  6. ajax 和xmlHttpRequest区别

    什么是 ajax ajax 即“Asynchronous JavaScript and XML”(异步 JavaScript 和 XML),也就是无刷新数据读取. http 请求 首先需要了解 htt ...

  7. C#委托举例

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  8. iOS菊花加载圈

    自定制一个继承于UIView的类然后重写initWithFrame方法;如下 - (id)initWithFrame:(CGRect)frame { self = [super initWithFra ...

  9. Android——监听事件OnLongClickListener

    .xml <Button android:layout_width="wrap_content" android:layout_height="wrap_conte ...

  10. java后台json如何传递到jsp中解析

    需求:  系统前端jsp使用的是easyUi的datagrid展示了一些任务信息,任务信息中有个状态信息显示的值是数字, 需要根据后台保存的映射关系,将状态显示为描述信息. 原来的jsp前端显示: 解 ...