Path: blob/master/modules/imgproc/src/featureselect.cpp
16354 views
/*M///////////////////////////////////////////////////////////////////////////////////////1//2// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.3//4// By downloading, copying, installing or using the software you agree to this license.5// If you do not agree to this license, do not download, install,6// copy or use the software.7//8//9// Intel License Agreement10// For Open Source Computer Vision Library11//12// Copyright (C) 2000, Intel Corporation, all rights reserved.13// Third party copyrights are property of their respective owners.14//15// Redistribution and use in source and binary forms, with or without modification,16// are permitted provided that the following conditions are met:17//18// * Redistribution's of source code must retain the above copyright notice,19// this list of conditions and the following disclaimer.20//21// * Redistribution's in binary form must reproduce the above copyright notice,22// this list of conditions and the following disclaimer in the documentation23// and/or other materials provided with the distribution.24//25// * The name of Intel Corporation may not be used to endorse or promote products26// derived from this software without specific prior written permission.27//28// This software is provided by the copyright holders and contributors "as is" and29// any express or implied warranties, including, but not limited to, the implied30// warranties of merchantability and fitness for a particular purpose are disclaimed.31// In no event shall the Intel Corporation or contributors be liable for any direct,32// indirect, incidental, special, exemplary, or consequential damages33// (including, but not limited to, procurement of substitute goods or services;34// loss of use, data, or profits; or business interruption) however caused35// and on any theory of liability, whether in contract, strict liability,36// or tort (including negligence or otherwise) arising in any way out of37// the use of this software, even if advised of the possibility of such damage.38//39//M*/4041#include "precomp.hpp"42#include "opencl_kernels_imgproc.hpp"4344#include "opencv2/core/openvx/ovx_defs.hpp"4546#include <cstdio>47#include <vector>48#include <iostream>49#include <functional>5051namespace cv52{5354struct greaterThanPtr55{56bool operator () (const float * a, const float * b) const57// Ensure a fully deterministic result of the sort58{ return (*a > *b) ? true : (*a < *b) ? false : (a > b); }59};6061#ifdef HAVE_OPENCL6263struct Corner64{65float val;66short y;67short x;6869bool operator < (const Corner & c) const70// Ensure a fully deterministic result of the sort71{ return (val > c.val) ? true : (val < c.val) ? false : (y > c.y) ? true : (y < c.y) ? false : (x > c.x); }72};7374static bool ocl_goodFeaturesToTrack( InputArray _image, OutputArray _corners,75int maxCorners, double qualityLevel, double minDistance,76InputArray _mask, int blockSize, int gradientSize,77bool useHarrisDetector, double harrisK )78{79UMat eig, maxEigenValue;80if( useHarrisDetector )81cornerHarris( _image, eig, blockSize, gradientSize, harrisK );82else83cornerMinEigenVal( _image, eig, blockSize, gradientSize );8485Size imgsize = _image.size();86size_t total, i, j, ncorners = 0, possibleCornersCount =87std::max(1024, static_cast<int>(imgsize.area() * 0.1));88bool haveMask = !_mask.empty();89UMat corners_buffer(1, (int)possibleCornersCount + 1, CV_32FC2);90CV_Assert(sizeof(Corner) == corners_buffer.elemSize());91Mat tmpCorners;9293// find threshold94{95CV_Assert(eig.type() == CV_32FC1);96int dbsize = ocl::Device::getDefault().maxComputeUnits();97size_t wgs = ocl::Device::getDefault().maxWorkGroupSize();9899int wgs2_aligned = 1;100while (wgs2_aligned < (int)wgs)101wgs2_aligned <<= 1;102wgs2_aligned >>= 1;103104ocl::Kernel k("maxEigenVal", ocl::imgproc::gftt_oclsrc,105format("-D OP_MAX_EIGEN_VAL -D WGS=%d -D groupnum=%d -D WGS2_ALIGNED=%d%s",106(int)wgs, dbsize, wgs2_aligned, haveMask ? " -D HAVE_MASK" : ""));107if (k.empty())108return false;109110UMat mask = _mask.getUMat();111maxEigenValue.create(1, dbsize, CV_32FC1);112113ocl::KernelArg eigarg = ocl::KernelArg::ReadOnlyNoSize(eig),114dbarg = ocl::KernelArg::PtrWriteOnly(maxEigenValue),115maskarg = ocl::KernelArg::ReadOnlyNoSize(mask),116cornersarg = ocl::KernelArg::PtrWriteOnly(corners_buffer);117118if (haveMask)119k.args(eigarg, eig.cols, (int)eig.total(), dbarg, maskarg);120else121k.args(eigarg, eig.cols, (int)eig.total(), dbarg);122123size_t globalsize = dbsize * wgs;124if (!k.run(1, &globalsize, &wgs, false))125return false;126127ocl::Kernel k2("maxEigenValTask", ocl::imgproc::gftt_oclsrc,128format("-D OP_MAX_EIGEN_VAL -D WGS=%zu -D WGS2_ALIGNED=%d -D groupnum=%d",129wgs, wgs2_aligned, dbsize));130if (k2.empty())131return false;132133k2.args(dbarg, (float)qualityLevel, cornersarg);134135if (!k2.runTask(false))136return false;137}138139// collect list of pointers to features - put them into temporary image140{141ocl::Kernel k("findCorners", ocl::imgproc::gftt_oclsrc,142format("-D OP_FIND_CORNERS%s", haveMask ? " -D HAVE_MASK" : ""));143if (k.empty())144return false;145146ocl::KernelArg eigarg = ocl::KernelArg::ReadOnlyNoSize(eig),147cornersarg = ocl::KernelArg::PtrWriteOnly(corners_buffer),148thresholdarg = ocl::KernelArg::PtrReadOnly(maxEigenValue);149150if (!haveMask)151k.args(eigarg, cornersarg, eig.rows - 2, eig.cols - 2, thresholdarg,152(int)possibleCornersCount);153else154{155UMat mask = _mask.getUMat();156k.args(eigarg, ocl::KernelArg::ReadOnlyNoSize(mask),157cornersarg, eig.rows - 2, eig.cols - 2,158thresholdarg, (int)possibleCornersCount);159}160161size_t globalsize[2] = { (size_t)eig.cols - 2, (size_t)eig.rows - 2 };162if (!k.run(2, globalsize, NULL, false))163return false;164165tmpCorners = corners_buffer.getMat(ACCESS_RW);166total = std::min<size_t>(tmpCorners.at<Vec2i>(0, 0)[0], possibleCornersCount);167if (total == 0)168{169_corners.release();170return true;171}172}173174Corner* corner_ptr = tmpCorners.ptr<Corner>() + 1;175std::sort(corner_ptr, corner_ptr + total);176177std::vector<Point2f> corners;178corners.reserve(total);179180if (minDistance >= 1)181{182// Partition the image into larger grids183int w = imgsize.width, h = imgsize.height;184185const int cell_size = cvRound(minDistance);186const int grid_width = (w + cell_size - 1) / cell_size;187const int grid_height = (h + cell_size - 1) / cell_size;188189std::vector<std::vector<Point2f> > grid(grid_width*grid_height);190minDistance *= minDistance;191192for( i = 0; i < total; i++ )193{194const Corner & c = corner_ptr[i];195bool good = true;196197int x_cell = c.x / cell_size;198int y_cell = c.y / cell_size;199200int x1 = x_cell - 1;201int y1 = y_cell - 1;202int x2 = x_cell + 1;203int y2 = y_cell + 1;204205// boundary check206x1 = std::max(0, x1);207y1 = std::max(0, y1);208x2 = std::min(grid_width - 1, x2);209y2 = std::min(grid_height - 1, y2);210211for( int yy = y1; yy <= y2; yy++ )212for( int xx = x1; xx <= x2; xx++ )213{214std::vector<Point2f> &m = grid[yy * grid_width + xx];215216if( m.size() )217{218for(j = 0; j < m.size(); j++)219{220float dx = c.x - m[j].x;221float dy = c.y - m[j].y;222223if( dx*dx + dy*dy < minDistance )224{225good = false;226goto break_out;227}228}229}230}231232break_out:233234if (good)235{236grid[y_cell*grid_width + x_cell].push_back(Point2f((float)c.x, (float)c.y));237238corners.push_back(Point2f((float)c.x, (float)c.y));239++ncorners;240241if( maxCorners > 0 && (int)ncorners == maxCorners )242break;243}244}245}246else247{248for( i = 0; i < total; i++ )249{250const Corner & c = corner_ptr[i];251252corners.push_back(Point2f((float)c.x, (float)c.y));253++ncorners;254if( maxCorners > 0 && (int)ncorners == maxCorners )255break;256}257}258259Mat(corners).convertTo(_corners, _corners.fixedType() ? _corners.type() : CV_32F);260return true;261}262263#endif264265#ifdef HAVE_OPENVX266struct VxKeypointsComparator267{268bool operator () (const vx_keypoint_t& a, const vx_keypoint_t& b)269{270return a.strength > b.strength;271}272};273274static bool openvx_harris(Mat image, OutputArray _corners,275int _maxCorners, double _qualityLevel, double _minDistance,276int _blockSize, int _gradientSize, double _harrisK)277{278using namespace ivx;279280if(image.type() != CV_8UC1) return false;281282//OpenVX implementations don't have to provide other sizes283if(!(_blockSize == 3 || _blockSize == 5 || _blockSize == 7)) return false;284285try286{287Context context = ovx::getOpenVXContext();288289Image ovxImage = Image::createFromHandle(context, Image::matTypeToFormat(image.type()),290Image::createAddressing(image), image.data);291//The minimum threshold which to eliminate Harris Corner scores (computed using the normalized Sobel kernel).292//set to 0, we'll filter it later by threshold293ivx::Scalar strengthThresh = ivx::Scalar::create<VX_TYPE_FLOAT32>(context, 0);294295//The gradient window size to use on the input.296vx_int32 gradientSize = _gradientSize;297298//The block window size used to compute the harris corner score299vx_int32 blockSize = _blockSize;300301//The scalar sensitivity threshold k from the Harris-Stephens equation302ivx::Scalar sensivity = ivx::Scalar::create<VX_TYPE_FLOAT32>(context, _harrisK);303304//The radial Euclidean distance for non-maximum suppression305ivx::Scalar minDistance = ivx::Scalar::create<VX_TYPE_FLOAT32>(context, _minDistance);306307vx_size capacity = image.cols * image.rows;308Array corners = Array::create(context, VX_TYPE_KEYPOINT, capacity);309ivx::Scalar numCorners = ivx::Scalar::create<VX_TYPE_SIZE>(context, 0);310311IVX_CHECK_STATUS(vxuHarrisCorners(context, ovxImage, strengthThresh, minDistance, sensivity,312gradientSize, blockSize, corners, numCorners));313314std::vector<vx_keypoint_t> vxKeypoints;315corners.copyTo(vxKeypoints);316317std::sort(vxKeypoints.begin(), vxKeypoints.end(), VxKeypointsComparator());318319vx_float32 maxStrength = 0.0f;320if(vxKeypoints.size() > 0)321maxStrength = vxKeypoints[0].strength;322size_t maxKeypoints = min((size_t)_maxCorners, vxKeypoints.size());323std::vector<Point2f> keypoints;324keypoints.reserve(maxKeypoints);325for(size_t i = 0; i < maxKeypoints; i++)326{327vx_keypoint_t kp = vxKeypoints[i];328if(kp.strength < maxStrength*_qualityLevel) break;329keypoints.push_back(Point2f((float)kp.x, (float)kp.y));330}331332Mat(keypoints).convertTo(_corners, _corners.fixedType() ? _corners.type() : CV_32F);333334#ifdef VX_VERSION_1_1335//we should take user memory back before release336//(it's not done automatically according to standard)337ovxImage.swapHandle();338#endif339}340catch (RuntimeError & e)341{342VX_DbgThrow(e.what());343}344catch (WrapperError & e)345{346VX_DbgThrow(e.what());347}348349return true;350}351352#endif353354}355356void cv::goodFeaturesToTrack( InputArray _image, OutputArray _corners,357int maxCorners, double qualityLevel, double minDistance,358InputArray _mask, int blockSize, int gradientSize,359bool useHarrisDetector, double harrisK )360{361CV_INSTRUMENT_REGION();362363CV_Assert( qualityLevel > 0 && minDistance >= 0 && maxCorners >= 0 );364CV_Assert( _mask.empty() || (_mask.type() == CV_8UC1 && _mask.sameSize(_image)) );365366CV_OCL_RUN(_image.dims() <= 2 && _image.isUMat(),367ocl_goodFeaturesToTrack(_image, _corners, maxCorners, qualityLevel, minDistance,368_mask, blockSize, gradientSize, useHarrisDetector, harrisK))369370Mat image = _image.getMat(), eig, tmp;371if (image.empty())372{373_corners.release();374return;375}376377// Disabled due to bad accuracy378CV_OVX_RUN(false && useHarrisDetector && _mask.empty() &&379!ovx::skipSmallImages<VX_KERNEL_HARRIS_CORNERS>(image.cols, image.rows),380openvx_harris(image, _corners, maxCorners, qualityLevel, minDistance, blockSize, gradientSize, harrisK))381382if( useHarrisDetector )383cornerHarris( image, eig, blockSize, gradientSize, harrisK );384else385cornerMinEigenVal( image, eig, blockSize, gradientSize );386387double maxVal = 0;388minMaxLoc( eig, 0, &maxVal, 0, 0, _mask );389threshold( eig, eig, maxVal*qualityLevel, 0, THRESH_TOZERO );390dilate( eig, tmp, Mat());391392Size imgsize = image.size();393std::vector<const float*> tmpCorners;394395// collect list of pointers to features - put them into temporary image396Mat mask = _mask.getMat();397for( int y = 1; y < imgsize.height - 1; y++ )398{399const float* eig_data = (const float*)eig.ptr(y);400const float* tmp_data = (const float*)tmp.ptr(y);401const uchar* mask_data = mask.data ? mask.ptr(y) : 0;402403for( int x = 1; x < imgsize.width - 1; x++ )404{405float val = eig_data[x];406if( val != 0 && val == tmp_data[x] && (!mask_data || mask_data[x]) )407tmpCorners.push_back(eig_data + x);408}409}410411std::vector<Point2f> corners;412size_t i, j, total = tmpCorners.size(), ncorners = 0;413414if (total == 0)415{416_corners.release();417return;418}419420std::sort( tmpCorners.begin(), tmpCorners.end(), greaterThanPtr() );421422if (minDistance >= 1)423{424// Partition the image into larger grids425int w = image.cols;426int h = image.rows;427428const int cell_size = cvRound(minDistance);429const int grid_width = (w + cell_size - 1) / cell_size;430const int grid_height = (h + cell_size - 1) / cell_size;431432std::vector<std::vector<Point2f> > grid(grid_width*grid_height);433434minDistance *= minDistance;435436for( i = 0; i < total; i++ )437{438int ofs = (int)((const uchar*)tmpCorners[i] - eig.ptr());439int y = (int)(ofs / eig.step);440int x = (int)((ofs - y*eig.step)/sizeof(float));441442bool good = true;443444int x_cell = x / cell_size;445int y_cell = y / cell_size;446447int x1 = x_cell - 1;448int y1 = y_cell - 1;449int x2 = x_cell + 1;450int y2 = y_cell + 1;451452// boundary check453x1 = std::max(0, x1);454y1 = std::max(0, y1);455x2 = std::min(grid_width-1, x2);456y2 = std::min(grid_height-1, y2);457458for( int yy = y1; yy <= y2; yy++ )459{460for( int xx = x1; xx <= x2; xx++ )461{462std::vector <Point2f> &m = grid[yy*grid_width + xx];463464if( m.size() )465{466for(j = 0; j < m.size(); j++)467{468float dx = x - m[j].x;469float dy = y - m[j].y;470471if( dx*dx + dy*dy < minDistance )472{473good = false;474goto break_out;475}476}477}478}479}480481break_out:482483if (good)484{485grid[y_cell*grid_width + x_cell].push_back(Point2f((float)x, (float)y));486487corners.push_back(Point2f((float)x, (float)y));488++ncorners;489490if( maxCorners > 0 && (int)ncorners == maxCorners )491break;492}493}494}495else496{497for( i = 0; i < total; i++ )498{499int ofs = (int)((const uchar*)tmpCorners[i] - eig.ptr());500int y = (int)(ofs / eig.step);501int x = (int)((ofs - y*eig.step)/sizeof(float));502503corners.push_back(Point2f((float)x, (float)y));504++ncorners;505if( maxCorners > 0 && (int)ncorners == maxCorners )506break;507}508}509510Mat(corners).convertTo(_corners, _corners.fixedType() ? _corners.type() : CV_32F);511}512513CV_IMPL void514cvGoodFeaturesToTrack( const void* _image, void*, void*,515CvPoint2D32f* _corners, int *_corner_count,516double quality_level, double min_distance,517const void* _maskImage, int block_size,518int use_harris, double harris_k )519{520cv::Mat image = cv::cvarrToMat(_image), mask;521std::vector<cv::Point2f> corners;522523if( _maskImage )524mask = cv::cvarrToMat(_maskImage);525526CV_Assert( _corners && _corner_count );527cv::goodFeaturesToTrack( image, corners, *_corner_count, quality_level,528min_distance, mask, block_size, use_harris != 0, harris_k );529530size_t i, ncorners = corners.size();531for( i = 0; i < ncorners; i++ )532_corners[i] = cvPoint2D32f(corners[i]);533*_corner_count = (int)ncorners;534}535536void cv::goodFeaturesToTrack( InputArray _image, OutputArray _corners,537int maxCorners, double qualityLevel, double minDistance,538InputArray _mask, int blockSize,539bool useHarrisDetector, double harrisK )540{541cv::goodFeaturesToTrack(_image, _corners, maxCorners, qualityLevel, minDistance,542_mask, blockSize, 3, useHarrisDetector, harrisK );543}544/* End of file. */545546547