HLS图像处理系列——肤色检測
本博文採用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图像处理系列——肤色检測的更多相关文章
- 图像处理之Canny边缘检測
图像处理之Canny 边缘检測 一:历史 Canny边缘检測算法是1986年有John F. Canny开发出来一种基于图像梯度计算的边缘 检測算法,同一时候Canny本人对计算图像边缘提取学科的发展 ...
- 爱国者布局智能硬件,空探系列PM2.5检測仪“嗅霾狗”大曝光
随着6月1日史上最严禁烟令的正式实施,国内包含北京.上海.成都等大中型城市已经在公共场所全面禁烟.众所周知,实施禁烟令的根本在于促进空气的净化,实现环境的改善,要达到这个目的,光有禁烟令是远远 ...
- Python图像处理(8):边缘检測
快乐虾 http://blog.csdn.net/lights_joy/ 欢迎转载,但请保留作者信息 此前已经得到了单个区域植株图像,接下来似乎应该尝试对这些区域进行分类识别.通过外形和叶脉进行植物种 ...
- OpenCV图像处理篇之边缘检測算子
3种边缘检測算子 灰度或结构等信息的突变位置是图像的边缘,图像的边缘有幅度和方向属性.沿边缘方向像素变化缓慢,垂直边缘方向像素变化剧烈.因此,边缘上的变化能通过梯度计算出来. 一阶导数的梯度算子 对于 ...
- 【OpenCV新手教程之十二】OpenCV边缘检測:Canny算子,Sobel算子,Laplace算子,Scharr滤波器合辑
本系列文章由@浅墨_毛星云 出品,转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/25560901 作者:毛星云(浅墨) ...
- Matlab图像处理系列2———空间域平滑滤波器
注:本系列来自于图像处理课程实验,用Matlab实现最主要的图像处理算法 本文章是Matlab图像处理系列的第二篇文章.介绍了空间域图像处理最主要的概念----模版和滤波器,给出了均值滤波起和中值滤波 ...
- 【OpenCV新手教程之十七】OpenCV重映射 & SURF特征点检測合辑
本系列文章由@浅墨_毛星云 出品.转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/30974513 作者:毛星云(浅墨) ...
- 目标检測的图像特征提取之(一)HOG特征
1.HOG特征: 方向梯度直方图(Histogram of Oriented Gradient, HOG)特征是一种在计算机视觉和图像处理中用来进行物体检測的特征描写叙述子.它通过计算和统计图像局部区 ...
- AIX下RAC搭建 Oracle10G(一)检測系统环境
AIX下RAC搭建系列 环境 节点 节点1 节点2 小机型号 IBM P-series 630 IBM P-series 630 主机名 AIX203 AIX204 交换机 SAN光纤交换机 存储 S ...
随机推荐
- Ribbon重试机制与Hystrix熔断机制的配置问题1
Ribbon超时与Hystrix超时问题,为了确保Ribbon重试的时候不被熔断,我们就需要让Hystrix的超时时间大于Ribbon的超时时间,否则Hystrix命令超时后,该命令直接熔断,重试机制 ...
- Python的sys.argv使用说明 通过终端写入环境变量
刚开始使用这个参数的时候,很不明白其含义.网上搜索很多都是贴的官网上面的一则实例,说看懂,就明白.可是,我看不懂.现在在回头看这个参数使用,并不是很麻烦. 举几个小例子就明白了. 创建一个脚本,内容如 ...
- poj 50道dp题
1.poj 3267 题意:给你一个字符串,下面有若干单词,问字符串要变成由下面单词组成的字符串,至少要删除多少个字母...... 例如: 6 10 browndcodw cow milk whit ...
- Tomcat Connector三种执行模式(BIO, NIO, APR)的比較和优化
Tomcat Connector的三种不同的执行模式性能相差非常大,有人測试过的结果例如以下: 这三种模式的不同之处例如以下: BIO: 一个线程处理一个请求.缺点:并发量高时,线程数较多,浪费资源. ...
- POST请求中,往URL传递数组
POST请求中,前端/客户端需要手动拼接URL,然后走接口跟后端交互. 若想传递数组,可以按照如下的拼接形式: &materialsTypeId=11&materialsTypeId= ...
- 【WPF】用代码给集合(Collection)容器动态添加子元素(Item)
需求:如何向 TabControl 中添加选项卡项. 问题:做的TabControl分页栏想要通过代码来控制添加的子元素.同理可以将解决思路拓展到用于其他的集合控件添加子元素的问题. 在布局文件She ...
- style="display:none"隐藏html的标签
隐藏html的标签 <div class="span11 alignment"> <h1>我的虚拟网络</h1> </div> &l ...
- jquery easy ui 简单字段选择搜索实现
code <!DOCTYPE html><html><head> <meta charset="UTF-8"> <title& ...
- 【R】提升R代码运算效率的11个实用方法
低.有许多种方法可以提升你的代码运算效率,但或许你更想了解运算效率能得到多大的提升.本文将介绍几种适用于大数据领域的方法,包括简单的逻辑调整设计.并行处理和Rcpp的运用,利用这些方法你可以轻松地处理 ...
- javaScript之function定义
背景知识 函数定义 在javaScript中,function的定义有3种: 1.匿名定义 function(){} 2.非匿名定义 fun ...