Opencv CUDA应用 图像峰值查找

图像处理中,经常通过距离变换对粘连物体图像进行分割。而距离变换常常通过峰值查找确定物体图像对质心。查找单通道图像局部峰值对方法,Matlab工具箱中有imregionalmax,python库skimage中feature.peak_local_max,但是OpenCV中没有相关查找局部峰值对函数。论坛中有人讨论使用滑动窗口,通过图像分割成几个局部,查找局部中对全局峰值来确定局部峰值,但是误差可想而知而且对于峰值出现为一个等值平面的话,是否能够找出峰值就取决于窗的大小。

论文《Morphological Grayscale Reconstruction in Image Analysis:
Applications and Effcient Algorithms》中提出了一种通过形态学灰度重建(Morphological Grayscale Reconstruction)对方法来求取局部峰值,Matlab中就是应用该方法实现得,详见imreconstruct说明中对参考文献,正是该篇论文。使用原图像数值偏移-1作为mask,对原图像进行形态学灰度重建,得到重建图。然后用原图像减去重建图就能得求出峰值的位置(大于零处)。


形态学灰度重建(图源于论文)

论文中多种形态学灰度重建算法,其中,运用并行计算的方法如下图。NG表示对应像素点对邻近像素,可取4邻近或8邻近。本文使用8邻近对算法进行实现。


并行算法实现(图源于论文)

使用CUDA对上述算法进行实现,对圆形粘连图像进行距离变换后查找峰值。(该算法的非CUDA实现,详见 【图像处理】灰度图、亮度峰值极值查找)图像效果如下:

原分割图像
距离变换
峰值查找

源代码:
imregionmax.cu

#include "cuda.h"
#include "cuda_runtime.h"
#include "cuda_runtime_api.h"
#include "device_functions.h"

#include "opencv2/core/core.hpp"
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/opencv.hpp"
#include "stdio.h"
#include <vector>
#include <iostream>
using namespace cv;
using namespace std;

#define Accuracy 0
typedef  unsigned char eleType;
__global__ void  DilationStep(eleType *k,eleType *j,unsigned int total)
{
    unsigned int x = blockIdx.x*blockDim.x + threadIdx.x;
    unsigned int y = blockIdx.y*blockDim.y + threadIdx.y;
    unsigned int offset = x + y*blockDim.x*gridDim.x;

    unsigned int width  = blockDim.x*gridDim.x;
    unsigned int heigth = blockDim.y*gridDim.y;

    if(offset > total) return;

    unsigned int left,right,top,bottom;
    left = offset -1;
    right = offset+1;
    if (x==0) left++;
    if (x==width-1) right--;
    top     = offset - width;
    bottom  = offset + width;
    if (y==0)           top += width;
    if (y==heigth-1)    bottom -= width;

    eleType max = j[offset];
    if(j[left]  -   max > Accuracy)     max = j[left];
    if(j[right] -   max > Accuracy)     max = j[right];
    if(j[bottom]-   max > Accuracy)     max = j[bottom];
    if(j[top]   -   max > Accuracy)     max = j[top];
    unsigned int leftbottom,lefttop,righttop,rightbottom;
    leftbottom  = bottom - 1;
    if(x==0) leftbottom++;
    rightbottom = bottom + 1;
    if(x==width-1) rightbottom--;
    lefttop     = top    - 1;
    if(x==0)    lefttop++;
    righttop    = top    + 1;
    if(x==width-1) righttop--;

    if(j[leftbottom]    -   max > Accuracy)     max = j[leftbottom];
    if(j[rightbottom]   -   max > Accuracy)     max = j[rightbottom];
    if(j[lefttop]       -   max > Accuracy)     max = j[lefttop];
    if(j[righttop]      -   max > Accuracy)     max = j[righttop];

    k[offset] =max;
}
__global__ void PointwiseMinimum(eleType *I,eleType *J,eleType *K,unsigned int total)
{
    unsigned int x = blockIdx.x*blockDim.x + threadIdx.x;
    unsigned int y = blockIdx.y*blockDim.y + threadIdx.y;
    unsigned int offset = x + y*blockDim.x*gridDim.x;
    if(I[offset] - K[offset] <Accuracy)
        J[offset] = I[offset];
    else
        J[offset] = K[offset];
}

#define DIM 16
Mat imregionmax(const Mat *src,eleType h)
{
     Mat LocMax     = src->clone();
     int width      = src->cols;
     int height     = src->rows;
     Mat Imask      = src->clone();
     Mat Jmasker    = Imask - h;
     Mat K          = Jmasker.clone();
     Mat Tmp        = src->clone();

     eleType *Jmasker_dev,*Imask_dev,*K_dev;

     cudaMalloc((void**)&Jmasker_dev,width*height*sizeof(eleType));
     cudaMemcpy(Jmasker_dev,Jmasker.data,width*height*sizeof(eleType),cudaMemcpyHostToDevice);
     cudaMalloc((void**)&Imask_dev,width*height*sizeof(eleType));
     cudaMemcpy(Imask_dev,Imask.data,width*height*sizeof(eleType),cudaMemcpyHostToDevice);
     cudaMalloc((void**)&K_dev,width*height*sizeof(eleType));

     cudaError_t cudaStatus;
     while(1)
     {

         dim3 blocks(64,64);
         dim3 threads((blocks.x+width-1)/blocks.x,(blocks.y+height-1)/blocks.y);
         DilationStep<<<blocks,threads>>>(K_dev,Jmasker_dev,width*height);
         cudaDeviceSynchronize();
         cudaStatus= cudaGetLastError();
         if (cudaStatus != cudaSuccess)
         {
             fprintf(stderr, "addKernel launch failed: %s\n", cudaGetErrorString(cudaStatus));
             return LocMax;
         }
         cudaMemcpy(K.data,K_dev,width*height*sizeof(eleType),cudaMemcpyDeviceToHost);
         PointwiseMinimum<<<blocks,threads>>>(Imask_dev,Jmasker_dev,K_dev,width*height);
         if (cudaStatus != cudaSuccess)
         {
             fprintf(stderr, "addKernel launch failed: %s\n", cudaGetErrorString(cudaStatus));
             return LocMax;
         }
         cudaMemcpy(Tmp.data,Jmasker_dev,width*height*sizeof(eleType),cudaMemcpyDeviceToHost);
         if (memcmp(Tmp.data,Jmasker.data,width*height*sizeof(eleType))==0) break;
         else cudaMemcpy(Jmasker.data,Jmasker_dev,width*height*sizeof(eleType),cudaMemcpyDeviceToHost);
     }
     cudaFree(Imask_dev);
     cudaFree(Jmasker_dev);
     cudaFree(K_dev);
     LocMax = (Imask-Jmasker>0);
     return LocMax;
}

main.cpp

#include <iostream>
#include "opencv2/core/core.hpp"
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/opencv.hpp"
using namespace std;
using namespace cv;
typedef  unsigned char eleType;
Mat imregionmax(const Mat *src,eleType h);

void LabShow(char wname[20],Mat src,Mat Label,Size offset)
{

    Mat sImg = src.clone();
    if(sImg.channels()<3)
        cvtColor(sImg,sImg,COLOR_GRAY2RGB);
    int font_face = cv::FONT_HERSHEY_COMPLEX;
    double font_scale = 1;
    for(int x=0;x<Label.cols;x++)
    {
        for(int y=0;y<Label.cols;y++)
        {
            Point pos;
            pos.x=x+offset.width;
            pos.y=y+offset.height;
            if(Label.at<uchar>(y,x))
                putText(sImg,"x",pos,font_face,font_scale,Scalar(0,0,255),3,8);
        }
    }
    imshow(wname,sImg);
}


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

    Mat src(Size(1024,1024),CV_8UC1);
    src.setTo(0);
    circle(src,Point(200,200),100,Scalar(255),-1);
    circle(src,Point(300,300),100,Scalar(255),-1);
    circle(src,Point(400,600),100,Scalar(255),-1);
    circle(src,Point(550,600),100,Scalar(255),-1);
    namedWindow("input", CV_WINDOW_NORMAL);
    imshow("input",src);

    Mat dis=Mat(src.size(),CV_32FC1);
    distanceTransform(src,dis,CV_DIST_L2,3);
    normalize(dis,dis,1.0,0.0,NORM_MINMAX);
    dis = dis *255;dis.convertTo(dis,CV_8U);
    namedWindow("distance Transfrom", CV_WINDOW_NORMAL);
    imshow("distance Transfrom",dis);

    namedWindow("peeks", CV_WINDOW_NORMAL);
    Mat peeks = imregionmax(&dis,1);
    LabShow("peeks",dis,peeks,Size(0,0));

    waitKey(0);
    return 0;
}

源码下载:
GitHub
参考:
Morphological grayscale reconstruction in image analysis: applications and efficient algorithms
python数字图像处理(19):骨架提取与分水岭算法

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容