mirror of https://github.com/opencv/opencv.git
Conflicts: .gitignore doc/tutorials/objdetect/cascade_classifier/cascade_classifier.rst modules/gpu/src/match_template.cpp modules/imgproc/include/opencv2/imgproc/imgproc.hpp modules/ocl/include/opencv2/ocl/ocl.hpp modules/ocl/perf/perf_precomp.hpppull/1427/head
commit
95c2e8b51f
41 changed files with 3519 additions and 148 deletions
@ -0,0 +1,282 @@ |
||||
/*M///////////////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.
|
||||
//
|
||||
// By downloading, copying, installing or using the software you agree to this license.
|
||||
// If you do not agree to this license, do not download, install,
|
||||
// copy or use the software.
|
||||
//
|
||||
//
|
||||
// License Agreement
|
||||
// For Open Source Computer Vision Library
|
||||
//
|
||||
// Copyright (C) 2010-2012, Multicoreware, Inc., all rights reserved.
|
||||
// Copyright (C) 2010-2012, Advanced Micro Devices, Inc., all rights reserved.
|
||||
// Third party copyrights are property of their respective owners.
|
||||
//
|
||||
// @Authors
|
||||
// Fangfang Bai, fangfang@multicorewareinc.com
|
||||
// Jin Ma, jin@multicorewareinc.com
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without modification,
|
||||
// are permitted provided that the following conditions are met:
|
||||
//
|
||||
// * Redistribution's of source code must retain the above copyright notice,
|
||||
// this list of conditions and the following disclaimer.
|
||||
//
|
||||
// * Redistribution's in binary form must reproduce the above copyright notice,
|
||||
// this list of conditions and the following disclaimer in the documentation
|
||||
// and/or other oclMaterials provided with the distribution.
|
||||
//
|
||||
// * The name of the copyright holders may not be used to endorse or promote products
|
||||
// derived from this software without specific prior written permission.
|
||||
//
|
||||
// This software is provided by the copyright holders and contributors as is and
|
||||
// any express or implied warranties, including, but not limited to, the implied
|
||||
// warranties of merchantability and fitness for a particular purpose are disclaimed.
|
||||
// In no event shall the Intel Corporation or contributors be liable for any direct,
|
||||
// indirect, incidental, special, exemplary, or consequential damages
|
||||
// (including, but not limited to, procurement of substitute goods or services;
|
||||
// loss of use, data, or profits; or business interruption) however caused
|
||||
// and on any theory of liability, whether in contract, strict liability,
|
||||
// or tort (including negligence or otherwise) arising in any way out of
|
||||
// the use of this software, even if advised of the possibility of such damage.
|
||||
//
|
||||
//M*/
|
||||
#include "perf_precomp.hpp" |
||||
using namespace perf; |
||||
using namespace std; |
||||
using namespace cv::ocl; |
||||
using namespace cv; |
||||
using std::tr1::tuple; |
||||
using std::tr1::get; |
||||
#if defined(HAVE_XINE) || \ |
||||
defined(HAVE_GSTREAMER) || \
|
||||
defined(HAVE_QUICKTIME) || \
|
||||
defined(HAVE_AVFOUNDATION) || \
|
||||
defined(HAVE_FFMPEG) || \
|
||||
defined(WIN32) |
||||
|
||||
# define BUILD_WITH_VIDEO_INPUT_SUPPORT 1 |
||||
#else |
||||
# define BUILD_WITH_VIDEO_INPUT_SUPPORT 0 |
||||
#endif |
||||
|
||||
#if BUILD_WITH_VIDEO_INPUT_SUPPORT |
||||
static void cvtFrameFmt(vector<Mat>& input, vector<Mat>& output) |
||||
{ |
||||
for(int i = 0; i< (int)(input.size()); i++) |
||||
{ |
||||
cvtColor(input[i], output[i], COLOR_RGB2GRAY); |
||||
} |
||||
} |
||||
//prepare data for CPU
|
||||
static void prepareData(VideoCapture& cap, int cn, vector<Mat>& frame_buffer) |
||||
{ |
||||
cv::Mat frame; |
||||
std::vector<Mat> frame_buffer_init; |
||||
int nFrame = (int)frame_buffer.size(); |
||||
for(int i = 0; i < nFrame; i++) |
||||
{ |
||||
cap >> frame; |
||||
ASSERT_FALSE(frame.empty()); |
||||
frame_buffer_init.push_back(frame); |
||||
} |
||||
|
||||
if(cn == 1) |
||||
cvtFrameFmt(frame_buffer_init, frame_buffer); |
||||
else |
||||
frame_buffer = frame_buffer_init; |
||||
} |
||||
//copy CPU data to GPU
|
||||
static void prepareData(vector<Mat>& frame_buffer, vector<oclMat>& frame_buffer_ocl) |
||||
{ |
||||
for(int i = 0; i < (int)frame_buffer.size(); i++) |
||||
frame_buffer_ocl.push_back(cv::ocl::oclMat(frame_buffer[i])); |
||||
} |
||||
#endif |
||||
///////////// MOG ////////////////////////
|
||||
#if BUILD_WITH_VIDEO_INPUT_SUPPORT |
||||
|
||||
typedef tuple<string, int, double> VideoMOGParamType; |
||||
typedef TestBaseWithParam<VideoMOGParamType> VideoMOGFixture; |
||||
|
||||
PERF_TEST_P(VideoMOGFixture, MOG, |
||||
::testing::Combine(::testing::Values("gpu/video/768x576.avi", "gpu/video/1920x1080.avi"), |
||||
::testing::Values(1, 3), |
||||
::testing::Values(0.0, 0.01))) |
||||
{ |
||||
VideoMOGParamType params = GetParam(); |
||||
|
||||
const string inputFile = perf::TestBase::getDataPath(get<0>(params)); |
||||
const int cn = get<1>(params); |
||||
const float learningRate = static_cast<float>(get<2>(params)); |
||||
|
||||
const int nFrame = 5; |
||||
|
||||
Mat foreground_cpu; |
||||
std::vector<Mat> frame_buffer(nFrame); |
||||
std::vector<oclMat> frame_buffer_ocl; |
||||
|
||||
cv::VideoCapture cap(inputFile); |
||||
ASSERT_TRUE(cap.isOpened()); |
||||
|
||||
prepareData(cap, cn, frame_buffer); |
||||
|
||||
cv::Mat foreground; |
||||
cv::ocl::oclMat foreground_d; |
||||
if(RUN_PLAIN_IMPL) |
||||
{ |
||||
TEST_CYCLE() |
||||
{ |
||||
cv::Ptr<cv::BackgroundSubtractorMOG> mog = createBackgroundSubtractorMOG(); |
||||
foreground.release(); |
||||
for (int i = 0; i < nFrame; i++) |
||||
{ |
||||
mog->apply(frame_buffer[i], foreground, learningRate); |
||||
} |
||||
} |
||||
SANITY_CHECK(foreground); |
||||
}else if(RUN_OCL_IMPL) |
||||
{ |
||||
prepareData(frame_buffer, frame_buffer_ocl); |
||||
CV_Assert((int)(frame_buffer_ocl.size()) == nFrame); |
||||
OCL_TEST_CYCLE() |
||||
{ |
||||
cv::ocl::MOG d_mog; |
||||
foreground_d.release(); |
||||
for (int i = 0; i < nFrame; ++i) |
||||
{ |
||||
d_mog(frame_buffer_ocl[i], foreground_d, learningRate); |
||||
} |
||||
} |
||||
foreground_d.download(foreground); |
||||
SANITY_CHECK(foreground); |
||||
}else |
||||
OCL_PERF_ELSE |
||||
} |
||||
#endif |
||||
|
||||
///////////// MOG2 ////////////////////////
|
||||
#if BUILD_WITH_VIDEO_INPUT_SUPPORT |
||||
|
||||
typedef tuple<string, int> VideoMOG2ParamType; |
||||
typedef TestBaseWithParam<VideoMOG2ParamType> VideoMOG2Fixture; |
||||
|
||||
PERF_TEST_P(VideoMOG2Fixture, MOG2, |
||||
::testing::Combine(::testing::Values("gpu/video/768x576.avi", "gpu/video/1920x1080.avi"), |
||||
::testing::Values(1, 3))) |
||||
{ |
||||
VideoMOG2ParamType params = GetParam(); |
||||
|
||||
const string inputFile = perf::TestBase::getDataPath(get<0>(params)); |
||||
const int cn = get<1>(params); |
||||
int nFrame = 5; |
||||
|
||||
std::vector<cv::Mat> frame_buffer(nFrame); |
||||
std::vector<cv::ocl::oclMat> frame_buffer_ocl; |
||||
|
||||
cv::VideoCapture cap(inputFile); |
||||
ASSERT_TRUE(cap.isOpened()); |
||||
prepareData(cap, cn, frame_buffer); |
||||
cv::Mat foreground; |
||||
cv::ocl::oclMat foreground_d; |
||||
|
||||
if(RUN_PLAIN_IMPL) |
||||
{ |
||||
TEST_CYCLE() |
||||
{ |
||||
cv::Ptr<cv::BackgroundSubtractorMOG2> mog2 = createBackgroundSubtractorMOG2(); |
||||
mog2->set("detectShadows", false); |
||||
foreground.release(); |
||||
|
||||
for (int i = 0; i < nFrame; i++) |
||||
{ |
||||
mog2->apply(frame_buffer[i], foreground); |
||||
} |
||||
} |
||||
SANITY_CHECK(foreground); |
||||
}else if(RUN_OCL_IMPL) |
||||
{ |
||||
prepareData(frame_buffer, frame_buffer_ocl); |
||||
CV_Assert((int)(frame_buffer_ocl.size()) == nFrame); |
||||
OCL_TEST_CYCLE() |
||||
{ |
||||
cv::ocl::MOG2 d_mog2; |
||||
foreground_d.release(); |
||||
for (int i = 0; i < nFrame; i++) |
||||
{ |
||||
d_mog2(frame_buffer_ocl[i], foreground_d); |
||||
} |
||||
} |
||||
foreground_d.download(foreground); |
||||
SANITY_CHECK(foreground); |
||||
}else |
||||
OCL_PERF_ELSE |
||||
} |
||||
#endif |
||||
|
||||
///////////// MOG2_GetBackgroundImage //////////////////
|
||||
#if BUILD_WITH_VIDEO_INPUT_SUPPORT |
||||
|
||||
typedef TestBaseWithParam<VideoMOG2ParamType> Video_MOG2GetBackgroundImage; |
||||
|
||||
PERF_TEST_P(Video_MOG2GetBackgroundImage, MOG2, |
||||
::testing::Combine(::testing::Values("gpu/video/768x576.avi", "gpu/video/1920x1080.avi"), |
||||
::testing::Values(3))) |
||||
{ |
||||
VideoMOG2ParamType params = GetParam(); |
||||
|
||||
const string inputFile = perf::TestBase::getDataPath(get<0>(params)); |
||||
const int cn = get<1>(params); |
||||
int nFrame = 5; |
||||
|
||||
std::vector<cv::Mat> frame_buffer(nFrame); |
||||
std::vector<cv::ocl::oclMat> frame_buffer_ocl; |
||||
|
||||
cv::VideoCapture cap(inputFile); |
||||
ASSERT_TRUE(cap.isOpened()); |
||||
|
||||
prepareData(cap, cn, frame_buffer); |
||||
|
||||
cv::Mat foreground; |
||||
cv::Mat background; |
||||
cv::ocl::oclMat foreground_d; |
||||
cv::ocl::oclMat background_d; |
||||
|
||||
if(RUN_PLAIN_IMPL) |
||||
{ |
||||
TEST_CYCLE() |
||||
{ |
||||
cv::Ptr<cv::BackgroundSubtractorMOG2> mog2 = createBackgroundSubtractorMOG2(); |
||||
mog2->set("detectShadows", false); |
||||
foreground.release(); |
||||
background.release(); |
||||
for (int i = 0; i < nFrame; i++) |
||||
{ |
||||
mog2->apply(frame_buffer[i], foreground); |
||||
} |
||||
mog2->getBackgroundImage(background); |
||||
} |
||||
SANITY_CHECK(background); |
||||
}else if(RUN_OCL_IMPL) |
||||
{ |
||||
prepareData(frame_buffer, frame_buffer_ocl); |
||||
CV_Assert((int)(frame_buffer_ocl.size()) == nFrame); |
||||
OCL_TEST_CYCLE() |
||||
{ |
||||
cv::ocl::MOG2 d_mog2; |
||||
foreground_d.release(); |
||||
background_d.release(); |
||||
for (int i = 0; i < nFrame; i++) |
||||
{ |
||||
d_mog2(frame_buffer_ocl[i], foreground_d); |
||||
} |
||||
d_mog2.getBackgroundImage(background_d); |
||||
} |
||||
background_d.download(background); |
||||
SANITY_CHECK(background); |
||||
}else |
||||
OCL_PERF_ELSE |
||||
} |
||||
#endif |
@ -0,0 +1,638 @@ |
||||
/*M///////////////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.
|
||||
//
|
||||
// By downloading, copying, installing or using the software you agree to this license.
|
||||
// If you do not agree to this license, do not download, install,
|
||||
// copy or use the software.
|
||||
//
|
||||
//
|
||||
// License Agreement
|
||||
// For Open Source Computer Vision Library
|
||||
//
|
||||
// Copyright (C) 2010-2013, Multicoreware, Inc., all rights reserved.
|
||||
// Copyright (C) 2010-2013, Advanced Micro Devices, Inc., all rights reserved.
|
||||
// Third party copyrights are property of their respective owners.
|
||||
//
|
||||
// @Authors
|
||||
// Jin Ma, jin@multicorewareinc.com
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without modification,
|
||||
// are permitted provided that the following conditions are met:
|
||||
//
|
||||
// * Redistribution's of source code must retain the above copyright notice,
|
||||
// this list of conditions and the following disclaimer.
|
||||
//
|
||||
// * Redistribution's in binary form must reproduce the above copyright notice,
|
||||
// this list of conditions and the following disclaimer in the documentation
|
||||
// and/or other oclMaterials provided with the distribution.
|
||||
//
|
||||
// * The name of the copyright holders may not be used to endorse or promote products
|
||||
// derived from this software without specific prior written permission.
|
||||
//
|
||||
// This software is provided by the copyright holders and contributors as is and
|
||||
// any express or implied warranties, including, but not limited to, the implied
|
||||
// warranties of merchantability and fitness for a particular purpose are disclaimed.
|
||||
// In no event shall the Intel Corporation or contributors be liable for any direct,
|
||||
// indirect, incidental, special, exemplary, or consequential damages
|
||||
// (including, but not limited to, procurement of substitute goods or services;
|
||||
// loss of use, data, or profits; or business interruption) however caused
|
||||
// and on any theory of liability, whether in contract, strict liability,
|
||||
// or tort (including negligence or otherwise) arising in any way out of
|
||||
// the use of this software, even if advised of the possibility of such damage.
|
||||
//
|
||||
//M*/
|
||||
|
||||
#include "precomp.hpp" |
||||
using namespace cv; |
||||
using namespace cv::ocl; |
||||
namespace cv |
||||
{ |
||||
namespace ocl |
||||
{ |
||||
extern const char* bgfg_mog; |
||||
|
||||
typedef struct _contant_struct |
||||
{ |
||||
cl_float c_Tb; |
||||
cl_float c_TB; |
||||
cl_float c_Tg; |
||||
cl_float c_varInit; |
||||
cl_float c_varMin; |
||||
cl_float c_varMax; |
||||
cl_float c_tau; |
||||
cl_uchar c_shadowVal; |
||||
}contant_struct; |
||||
|
||||
cl_mem cl_constants = NULL; |
||||
float c_TB; |
||||
} |
||||
} |
||||
|
||||
#if defined _MSC_VER |
||||
#define snprintf sprintf_s |
||||
#endif |
||||
|
||||
namespace cv { namespace ocl { namespace device |
||||
{ |
||||
namespace mog |
||||
{ |
||||
void mog_ocl(const oclMat& frame, int cn, oclMat& fgmask, oclMat& weight, oclMat& sortKey, oclMat& mean, oclMat& var, |
||||
int nmixtures, float varThreshold, float learningRate, float backgroundRatio, float noiseSigma); |
||||
|
||||
void getBackgroundImage_ocl(int cn, const oclMat& weight, const oclMat& mean, oclMat& dst, int nmixtures, float backgroundRatio); |
||||
|
||||
void loadConstants(float Tb, float TB, float Tg, float varInit, float varMin, float varMax, float tau, |
||||
unsigned char shadowVal); |
||||
|
||||
void mog2_ocl(const oclMat& frame, int cn, oclMat& fgmask, oclMat& modesUsed, oclMat& weight, oclMat& variance, oclMat& mean, |
||||
float alphaT, float prune, bool detectShadows, int nmixtures); |
||||
|
||||
void getBackgroundImage2_ocl(int cn, const oclMat& modesUsed, const oclMat& weight, const oclMat& mean, oclMat& dst, int nmixtures); |
||||
} |
||||
}}} |
||||
|
||||
namespace mog |
||||
{ |
||||
const int defaultNMixtures = 5; |
||||
const int defaultHistory = 200; |
||||
const float defaultBackgroundRatio = 0.7f; |
||||
const float defaultVarThreshold = 2.5f * 2.5f; |
||||
const float defaultNoiseSigma = 30.0f * 0.5f; |
||||
const float defaultInitialWeight = 0.05f; |
||||
} |
||||
void cv::ocl::BackgroundSubtractor::operator()(const oclMat&, oclMat&, float) |
||||
{ |
||||
|
||||
} |
||||
cv::ocl::BackgroundSubtractor::~BackgroundSubtractor() |
||||
{ |
||||
|
||||
} |
||||
|
||||
cv::ocl::MOG::MOG(int nmixtures) : |
||||
frameSize_(0, 0), frameType_(0), nframes_(0) |
||||
{ |
||||
nmixtures_ = std::min(nmixtures > 0 ? nmixtures : mog::defaultNMixtures, 8); |
||||
history = mog::defaultHistory; |
||||
varThreshold = mog::defaultVarThreshold; |
||||
backgroundRatio = mog::defaultBackgroundRatio; |
||||
noiseSigma = mog::defaultNoiseSigma; |
||||
} |
||||
|
||||
void cv::ocl::MOG::initialize(cv::Size frameSize, int frameType) |
||||
{ |
||||
CV_Assert(frameType == CV_8UC1 || frameType == CV_8UC3 || frameType == CV_8UC4); |
||||
|
||||
frameSize_ = frameSize; |
||||
frameType_ = frameType; |
||||
|
||||
int ch = CV_MAT_CN(frameType); |
||||
int work_ch = ch; |
||||
|
||||
// for each gaussian mixture of each pixel bg model we store
|
||||
// the mixture sort key (w/sum_of_variances), the mixture weight (w),
|
||||
// the mean (nchannels values) and
|
||||
// the diagonal covariance matrix (another nchannels values)
|
||||
|
||||
weight_.create(frameSize.height * nmixtures_, frameSize_.width, CV_32FC1); |
||||
sortKey_.create(frameSize.height * nmixtures_, frameSize_.width, CV_32FC1); |
||||
mean_.create(frameSize.height * nmixtures_, frameSize_.width, CV_32FC(work_ch)); |
||||
var_.create(frameSize.height * nmixtures_, frameSize_.width, CV_32FC(work_ch)); |
||||
|
||||
weight_.setTo(cv::Scalar::all(0)); |
||||
sortKey_.setTo(cv::Scalar::all(0)); |
||||
mean_.setTo(cv::Scalar::all(0)); |
||||
var_.setTo(cv::Scalar::all(0)); |
||||
|
||||
nframes_ = 0; |
||||
} |
||||
|
||||
void cv::ocl::MOG::operator()(const cv::ocl::oclMat& frame, cv::ocl::oclMat& fgmask, float learningRate) |
||||
{ |
||||
using namespace cv::ocl::device::mog; |
||||
|
||||
CV_Assert(frame.depth() == CV_8U); |
||||
|
||||
int ch = frame.oclchannels(); |
||||
int work_ch = ch; |
||||
|
||||
if (nframes_ == 0 || learningRate >= 1.0 || frame.size() != frameSize_ || work_ch != mean_.oclchannels()) |
||||
initialize(frame.size(), frame.type()); |
||||
|
||||
fgmask.create(frameSize_, CV_8UC1); |
||||
|
||||
++nframes_; |
||||
learningRate = learningRate >= 0.0f && nframes_ > 1 ? learningRate : 1.0f / std::min(nframes_, history); |
||||
CV_Assert(learningRate >= 0.0f); |
||||
|
||||
mog_ocl(frame, ch, fgmask, weight_, sortKey_, mean_, var_, nmixtures_, |
||||
varThreshold, learningRate, backgroundRatio, noiseSigma); |
||||
} |
||||
|
||||
void cv::ocl::MOG::getBackgroundImage(oclMat& backgroundImage) const |
||||
{ |
||||
using namespace cv::ocl::device::mog; |
||||
|
||||
backgroundImage.create(frameSize_, frameType_); |
||||
|
||||
cv::ocl::device::mog::getBackgroundImage_ocl(backgroundImage.oclchannels(), weight_, mean_, backgroundImage, nmixtures_, backgroundRatio); |
||||
} |
||||
|
||||
void cv::ocl::MOG::release() |
||||
{ |
||||
frameSize_ = Size(0, 0); |
||||
frameType_ = 0; |
||||
nframes_ = 0; |
||||
|
||||
weight_.release(); |
||||
sortKey_.release(); |
||||
mean_.release(); |
||||
var_.release(); |
||||
clReleaseMemObject(cl_constants); |
||||
} |
||||
|
||||
static void mog_withoutLearning(const oclMat& frame, int cn, oclMat& fgmask, oclMat& weight, oclMat& mean, oclMat& var, |
||||
int nmixtures, float varThreshold, float backgroundRatio) |
||||
{ |
||||
Context* clCxt = Context::getContext(); |
||||
|
||||
size_t local_thread[] = {32, 8, 1}; |
||||
size_t global_thread[] = {frame.cols, frame.rows, 1}; |
||||
|
||||
int frame_step = (int)(frame.step/frame.elemSize()); |
||||
int fgmask_step = (int)(fgmask.step/fgmask.elemSize()); |
||||
int weight_step = (int)(weight.step/weight.elemSize()); |
||||
int mean_step = (int)(mean.step/mean.elemSize()); |
||||
int var_step = (int)(var.step/var.elemSize()); |
||||
|
||||
int fgmask_offset_y = (int)(fgmask.offset/fgmask.step); |
||||
int fgmask_offset_x = (int)(fgmask.offset%fgmask.step); |
||||
fgmask_offset_x = fgmask_offset_x/(int)fgmask.elemSize(); |
||||
|
||||
int frame_offset_y = (int)(frame.offset/frame.step); |
||||
int frame_offset_x = (int)(frame.offset%frame.step); |
||||
frame_offset_x = frame_offset_x/(int)frame.elemSize(); |
||||
|
||||
char build_option[50]; |
||||
if(cn == 1) |
||||
{ |
||||
snprintf(build_option, 50, "-D CN1 -D NMIXTURES=%d", nmixtures); |
||||
}else |
||||
{ |
||||
snprintf(build_option, 50, "-D NMIXTURES=%d", nmixtures); |
||||
} |
||||
|
||||
String kernel_name = "mog_withoutLearning_kernel"; |
||||
std::vector<std::pair<size_t, const void*> > args; |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&frame.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&fgmask.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&weight.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&mean.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&var.data)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame.rows)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame.cols)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&weight_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&mean_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&var_step)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&varThreshold)); |
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&backgroundRatio)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_offset_x)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_offset_y)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_offset_x)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_offset_y)); |
||||
|
||||
openCLExecuteKernel(clCxt, &bgfg_mog, kernel_name, global_thread, local_thread, args, -1, -1, build_option); |
||||
} |
||||
|
||||
|
||||
static void mog_withLearning(const oclMat& frame, int cn, oclMat& fgmask_raw, oclMat& weight, oclMat& sortKey, oclMat& mean, oclMat& var, |
||||
int nmixtures, float varThreshold, float backgroundRatio, float learningRate, float minVar) |
||||
{ |
||||
Context* clCxt = Context::getContext(); |
||||
|
||||
size_t local_thread[] = {32, 8, 1}; |
||||
size_t global_thread[] = {frame.cols, frame.rows, 1}; |
||||
|
||||
oclMat fgmask(fgmask_raw.size(), CV_32SC1); |
||||
|
||||
int frame_step = (int)(frame.step/frame.elemSize()); |
||||
int fgmask_step = (int)(fgmask.step/fgmask.elemSize()); |
||||
int weight_step = (int)(weight.step/weight.elemSize()); |
||||
int sortKey_step = (int)(sortKey.step/sortKey.elemSize()); |
||||
int mean_step = (int)(mean.step/mean.elemSize()); |
||||
int var_step = (int)(var.step/var.elemSize()); |
||||
|
||||
int fgmask_offset_y = (int)(fgmask.offset/fgmask.step); |
||||
int fgmask_offset_x = (int)(fgmask.offset%fgmask.step); |
||||
fgmask_offset_x = fgmask_offset_x/(int)fgmask.elemSize(); |
||||
|
||||
int frame_offset_y = (int)(frame.offset/frame.step); |
||||
int frame_offset_x = (int)(frame.offset%frame.step); |
||||
frame_offset_x = frame_offset_x/(int)frame.elemSize(); |
||||
|
||||
char build_option[50]; |
||||
if(cn == 1) |
||||
{ |
||||
snprintf(build_option, 50, "-D CN1 -D NMIXTURES=%d", nmixtures); |
||||
}else |
||||
{ |
||||
snprintf(build_option, 50, "-D NMIXTURES=%d", nmixtures); |
||||
} |
||||
|
||||
String kernel_name = "mog_withLearning_kernel"; |
||||
std::vector<std::pair<size_t, const void*> > args; |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&frame.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&fgmask.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&weight.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&sortKey.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&mean.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&var.data)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame.rows)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame.cols)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&weight_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&sortKey_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&mean_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&var_step)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&varThreshold)); |
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&backgroundRatio)); |
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&learningRate)); |
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&minVar)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_offset_x)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_offset_y)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_offset_x)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_offset_y)); |
||||
|
||||
openCLExecuteKernel(clCxt, &bgfg_mog, kernel_name, global_thread, local_thread, args, -1, -1, build_option); |
||||
fgmask.convertTo(fgmask, CV_8U); |
||||
fgmask.copyTo(fgmask_raw); |
||||
} |
||||
|
||||
void cv::ocl::device::mog::mog_ocl(const oclMat& frame, int cn, oclMat& fgmask, oclMat& weight, oclMat& sortKey, oclMat& mean, oclMat& var, |
||||
int nmixtures, float varThreshold, float learningRate, float backgroundRatio, float noiseSigma) |
||||
{ |
||||
const float minVar = noiseSigma * noiseSigma; |
||||
|
||||
if(learningRate > 0.0f) |
||||
mog_withLearning(frame, cn, fgmask, weight, sortKey, mean, var, nmixtures, |
||||
varThreshold, backgroundRatio, learningRate, minVar); |
||||
else |
||||
mog_withoutLearning(frame, cn, fgmask, weight, mean, var, nmixtures, varThreshold, backgroundRatio); |
||||
} |
||||
|
||||
void cv::ocl::device::mog::getBackgroundImage_ocl(int cn, const oclMat& weight, const oclMat& mean, oclMat& dst, int nmixtures, float backgroundRatio) |
||||
{ |
||||
Context* clCxt = Context::getContext(); |
||||
|
||||
size_t local_thread[] = {32, 8, 1}; |
||||
size_t global_thread[] = {dst.cols, dst.rows, 1}; |
||||
|
||||
int weight_step = (int)(weight.step/weight.elemSize()); |
||||
int mean_step = (int)(mean.step/mean.elemSize()); |
||||
int dst_step = (int)(dst.step/dst.elemSize()); |
||||
|
||||
char build_option[50]; |
||||
if(cn == 1) |
||||
{ |
||||
snprintf(build_option, 50, "-D CN1 -D NMIXTURES=%d", nmixtures); |
||||
}else |
||||
{ |
||||
snprintf(build_option, 50, "-D NMIXTURES=%d", nmixtures); |
||||
} |
||||
|
||||
String kernel_name = "getBackgroundImage_kernel"; |
||||
std::vector<std::pair<size_t, const void*> > args; |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&weight.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&mean.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&dst.data)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&dst.rows)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&dst.cols)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&weight_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&mean_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&dst_step)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&backgroundRatio)); |
||||
|
||||
openCLExecuteKernel(clCxt, &bgfg_mog, kernel_name, global_thread, local_thread, args, -1, -1, build_option); |
||||
} |
||||
|
||||
void cv::ocl::device::mog::loadConstants(float Tb, float TB, float Tg, float varInit, float varMin, float varMax, float tau, unsigned char shadowVal) |
||||
{ |
||||
varMin = cv::min(varMin, varMax); |
||||
varMax = cv::max(varMin, varMax); |
||||
|
||||
c_TB = TB; |
||||
|
||||
_contant_struct *constants = new _contant_struct; |
||||
constants->c_Tb = Tb; |
||||
constants->c_TB = TB; |
||||
constants->c_Tg = Tg; |
||||
constants->c_varInit = varInit; |
||||
constants->c_varMin = varMin; |
||||
constants->c_varMax = varMax; |
||||
constants->c_tau = tau; |
||||
constants->c_shadowVal = shadowVal; |
||||
|
||||
cl_constants = load_constant(*((cl_context*)getoclContext()), *((cl_command_queue*)getoclCommandQueue()), |
||||
(void *)constants, sizeof(_contant_struct)); |
||||
} |
||||
|
||||
void cv::ocl::device::mog::mog2_ocl(const oclMat& frame, int cn, oclMat& fgmaskRaw, oclMat& modesUsed, oclMat& weight, oclMat& variance, |
||||
oclMat& mean, float alphaT, float prune, bool detectShadows, int nmixtures) |
||||
{ |
||||
oclMat fgmask(fgmaskRaw.size(), CV_32SC1); |
||||
|
||||
Context* clCxt = Context::getContext(); |
||||
|
||||
const float alpha1 = 1.0f - alphaT; |
||||
|
||||
cl_int detectShadows_flag = 0; |
||||
if(detectShadows) |
||||
detectShadows_flag = 1; |
||||
|
||||
size_t local_thread[] = {32, 8, 1}; |
||||
size_t global_thread[] = {frame.cols, frame.rows, 1}; |
||||
|
||||
int frame_step = (int)(frame.step/frame.elemSize()); |
||||
int fgmask_step = (int)(fgmask.step/fgmask.elemSize()); |
||||
int weight_step = (int)(weight.step/weight.elemSize()); |
||||
int modesUsed_step = (int)(modesUsed.step/modesUsed.elemSize()); |
||||
int mean_step = (int)(mean.step/mean.elemSize()); |
||||
int var_step = (int)(variance.step/variance.elemSize()); |
||||
|
||||
int fgmask_offset_y = (int)(fgmask.offset/fgmask.step); |
||||
int fgmask_offset_x = (int)(fgmask.offset%fgmask.step); |
||||
fgmask_offset_x = fgmask_offset_x/(int)fgmask.elemSize(); |
||||
|
||||
int frame_offset_y = (int)(frame.offset/frame.step); |
||||
int frame_offset_x = (int)(frame.offset%frame.step); |
||||
frame_offset_x = frame_offset_x/(int)frame.elemSize(); |
||||
|
||||
String kernel_name = "mog2_kernel"; |
||||
std::vector<std::pair<size_t, const void*> > args; |
||||
|
||||
char build_option[50]; |
||||
if(cn == 1) |
||||
{ |
||||
snprintf(build_option, 50, "-D CN1 -D NMIXTURES=%d", nmixtures); |
||||
}else |
||||
{ |
||||
snprintf(build_option, 50, "-D NMIXTURES=%d", nmixtures); |
||||
} |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&frame.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&fgmask.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&weight.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&mean.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&modesUsed.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&variance.data)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame.rows)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame.cols)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&weight_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&mean_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&modesUsed_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&var_step)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&alphaT)); |
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&alpha1)); |
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&prune)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&detectShadows_flag)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_offset_x)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&fgmask_offset_y)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_offset_x)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&frame_offset_y)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&cl_constants)); |
||||
|
||||
openCLExecuteKernel(clCxt, &bgfg_mog, kernel_name, global_thread, local_thread, args, -1, -1, build_option); |
||||
|
||||
fgmask.convertTo(fgmask, CV_8U); |
||||
fgmask.copyTo(fgmaskRaw); |
||||
} |
||||
|
||||
void cv::ocl::device::mog::getBackgroundImage2_ocl(int cn, const oclMat& modesUsed, const oclMat& weight, const oclMat& mean, oclMat& dst, int nmixtures) |
||||
{ |
||||
Context* clCxt = Context::getContext(); |
||||
|
||||
size_t local_thread[] = {32, 8, 1}; |
||||
size_t global_thread[] = {modesUsed.cols, modesUsed.rows, 1}; |
||||
|
||||
int weight_step = (int)(weight.step/weight.elemSize()); |
||||
int modesUsed_step = (int)(modesUsed.step/modesUsed.elemSize()); |
||||
int mean_step = (int)(mean.step/mean.elemSize()); |
||||
int dst_step = (int)(dst.step/dst.elemSize()); |
||||
|
||||
int dst_y = (int)(dst.offset/dst.step); |
||||
int dst_x = (int)(dst.offset%dst.step); |
||||
dst_x = dst_x/(int)dst.elemSize(); |
||||
|
||||
String kernel_name = "getBackgroundImage2_kernel"; |
||||
std::vector<std::pair<size_t, const void*> > args; |
||||
|
||||
char build_option[50]; |
||||
if(cn == 1) |
||||
{ |
||||
snprintf(build_option, 50, "-D CN1 -D NMIXTURES=%d", nmixtures); |
||||
}else |
||||
{ |
||||
snprintf(build_option, 50, "-D NMIXTURES=%d", nmixtures); |
||||
} |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&modesUsed.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&weight.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&mean.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_mem), (void*)&dst.data)); |
||||
args.push_back(std::make_pair(sizeof(cl_float), (void*)&c_TB)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&modesUsed.rows)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&modesUsed.cols)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&modesUsed_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&weight_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&mean_step)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&dst_step)); |
||||
|
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&dst_x)); |
||||
args.push_back(std::make_pair(sizeof(cl_int), (void*)&dst_y)); |
||||
|
||||
openCLExecuteKernel(clCxt, &bgfg_mog, kernel_name, global_thread, local_thread, args, -1, -1, build_option); |
||||
} |
||||
|
||||
/////////////////////////////////////////////////////////////////
|
||||
// MOG2
|
||||
|
||||
namespace mog2 |
||||
{ |
||||
// default parameters of gaussian background detection algorithm
|
||||
const int defaultHistory = 500; // Learning rate; alpha = 1/defaultHistory2
|
||||
const float defaultVarThreshold = 4.0f * 4.0f; |
||||
const int defaultNMixtures = 5; // maximal number of Gaussians in mixture
|
||||
const float defaultBackgroundRatio = 0.9f; // threshold sum of weights for background test
|
||||
const float defaultVarThresholdGen = 3.0f * 3.0f; |
||||
const float defaultVarInit = 15.0f; // initial variance for new components
|
||||
const float defaultVarMax = 5.0f * defaultVarInit; |
||||
const float defaultVarMin = 4.0f; |
||||
|
||||
// additional parameters
|
||||
const float defaultfCT = 0.05f; // complexity reduction prior constant 0 - no reduction of number of components
|
||||
const unsigned char defaultnShadowDetection = 127; // value to use in the segmentation mask for shadows, set 0 not to do shadow detection
|
||||
const float defaultfTau = 0.5f; // Tau - shadow threshold, see the paper for explanation
|
||||
} |
||||
|
||||
cv::ocl::MOG2::MOG2(int nmixtures) : frameSize_(0, 0), frameType_(0), nframes_(0) |
||||
{ |
||||
nmixtures_ = nmixtures > 0 ? nmixtures : mog2::defaultNMixtures; |
||||
|
||||
history = mog2::defaultHistory; |
||||
varThreshold = mog2::defaultVarThreshold; |
||||
bShadowDetection = true; |
||||
|
||||
backgroundRatio = mog2::defaultBackgroundRatio; |
||||
fVarInit = mog2::defaultVarInit; |
||||
fVarMax = mog2::defaultVarMax; |
||||
fVarMin = mog2::defaultVarMin; |
||||
|
||||
varThresholdGen = mog2::defaultVarThresholdGen; |
||||
fCT = mog2::defaultfCT; |
||||
nShadowDetection = mog2::defaultnShadowDetection; |
||||
fTau = mog2::defaultfTau; |
||||
} |
||||
|
||||
void cv::ocl::MOG2::initialize(cv::Size frameSize, int frameType) |
||||
{ |
||||
using namespace cv::ocl::device::mog; |
||||
CV_Assert(frameType == CV_8UC1 || frameType == CV_8UC3 || frameType == CV_8UC4); |
||||
|
||||
frameSize_ = frameSize; |
||||
frameType_ = frameType; |
||||
nframes_ = 0; |
||||
|
||||
int ch = CV_MAT_CN(frameType); |
||||
int work_ch = ch; |
||||
|
||||
// for each gaussian mixture of each pixel bg model we store ...
|
||||
// the mixture weight (w),
|
||||
// the mean (nchannels values) and
|
||||
// the covariance
|
||||
weight_.create(frameSize.height * nmixtures_, frameSize_.width, CV_32FC1); |
||||
weight_.setTo(Scalar::all(0)); |
||||
|
||||
variance_.create(frameSize.height * nmixtures_, frameSize_.width, CV_32FC1); |
||||
variance_.setTo(Scalar::all(0)); |
||||
|
||||
mean_.create(frameSize.height * nmixtures_, frameSize_.width, CV_32FC(work_ch)); //4 channels
|
||||
mean_.setTo(Scalar::all(0)); |
||||
|
||||
//make the array for keeping track of the used modes per pixel - all zeros at start
|
||||
bgmodelUsedModes_.create(frameSize_, CV_32FC1); |
||||
bgmodelUsedModes_.setTo(cv::Scalar::all(0)); |
||||
|
||||
loadConstants(varThreshold, backgroundRatio, varThresholdGen, fVarInit, fVarMin, fVarMax, fTau, nShadowDetection); |
||||
} |
||||
|
||||
void cv::ocl::MOG2::operator()(const oclMat& frame, oclMat& fgmask, float learningRate) |
||||
{ |
||||
using namespace cv::ocl::device::mog; |
||||
|
||||
int ch = frame.oclchannels(); |
||||
int work_ch = ch; |
||||
|
||||
if (nframes_ == 0 || learningRate >= 1.0f || frame.size() != frameSize_ || work_ch != mean_.oclchannels()) |
||||
initialize(frame.size(), frame.type()); |
||||
|
||||
fgmask.create(frameSize_, CV_8UC1); |
||||
fgmask.setTo(cv::Scalar::all(0)); |
||||
|
||||
++nframes_; |
||||
learningRate = learningRate >= 0.0f && nframes_ > 1 ? learningRate : 1.0f / std::min(2 * nframes_, history); |
||||
CV_Assert(learningRate >= 0.0f); |
||||
|
||||
mog2_ocl(frame, frame.oclchannels(), fgmask, bgmodelUsedModes_, weight_, variance_, mean_, learningRate, -learningRate * fCT, bShadowDetection, nmixtures_); |
||||
} |
||||
|
||||
void cv::ocl::MOG2::getBackgroundImage(oclMat& backgroundImage) const |
||||
{ |
||||
using namespace cv::ocl::device::mog; |
||||
|
||||
backgroundImage.create(frameSize_, frameType_); |
||||
|
||||
cv::ocl::device::mog::getBackgroundImage2_ocl(backgroundImage.oclchannels(), bgmodelUsedModes_, weight_, mean_, backgroundImage, nmixtures_); |
||||
} |
||||
|
||||
void cv::ocl::MOG2::release() |
||||
{ |
||||
frameSize_ = Size(0, 0); |
||||
frameType_ = 0; |
||||
nframes_ = 0; |
||||
|
||||
weight_.release(); |
||||
variance_.release(); |
||||
mean_.release(); |
||||
|
||||
bgmodelUsedModes_.release(); |
||||
} |
@ -0,0 +1,535 @@ |
||||
/*M/////////////////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING. |
||||
// |
||||
// By downloading, copying, installing or using the software you agree to this license. |
||||
// If you do not agree to this license, do not download, install, |
||||
// copy or use the software. |
||||
// |
||||
// |
||||
// License Agreement |
||||
// For Open Source Computer Vision Library |
||||
// |
||||
// Copyright (C) 2010-2013, Multicoreware, Inc., all rights reserved. |
||||
// Copyright (C) 2010-2013, Advanced Micro Devices, Inc., all rights reserved. |
||||
// Third party copyrights are property of their respective owners. |
||||
// |
||||
// @Authors |
||||
// Jin Ma jin@multicorewareinc.com |
||||
// |
||||
// Redistribution and use in source and binary forms, with or without modification, |
||||
// are permitted provided that the following conditions are met: |
||||
// |
||||
// * Redistribution's of source code must retain the above copyright notice, |
||||
// this list of conditions and the following disclaimer. |
||||
// |
||||
// * Redistribution's in binary form must reproduce the above copyright notice, |
||||
// this list of conditions and the following disclaimer in the documentation |
||||
// and/or other oclMaterials provided with the distribution. |
||||
// |
||||
// * The name of the copyright holders may not be used to endorse or promote products |
||||
// derived from this software without specific prior written permission. |
||||
// |
||||
// This software is provided by the copyright holders and contributors as is and |
||||
// any express or implied warranties, including, but not limited to, the implied |
||||
// warranties of merchantability and fitness for a particular purpose are disclaimed. |
||||
// In no event shall the Intel Corporation or contributors be liable for any direct, |
||||
// indirect, incidental, special, exemplary, or consequential damages |
||||
// (including, but not limited to, procurement of substitute goods or services; |
||||
// loss of use, data, or profits; or business interruption) however caused |
||||
// and on any theory of liability, whether in contract, strict liability, |
||||
// or tort (including negligence or otherwise) arising in any way out of |
||||
// the use of this software, even if advised of the possibility of such damage. |
||||
// |
||||
//M*/ |
||||
|
||||
#if defined (CN1) |
||||
#define T_FRAME uchar |
||||
#define T_MEAN_VAR float |
||||
#define CONVERT_TYPE convert_uchar_sat |
||||
#define F_ZERO (0.0f) |
||||
float cvt(uchar val) |
||||
{ |
||||
return val; |
||||
} |
||||
|
||||
float sqr(float val) |
||||
{ |
||||
return val * val; |
||||
} |
||||
|
||||
float sum(float val) |
||||
{ |
||||
return val; |
||||
} |
||||
|
||||
float clamp1(float var, float learningRate, float diff, float minVar) |
||||
{ |
||||
return fmax(var + learningRate * (diff * diff - var), minVar); |
||||
} |
||||
#else |
||||
#define T_FRAME uchar4 |
||||
#define T_MEAN_VAR float4 |
||||
#define CONVERT_TYPE convert_uchar4_sat |
||||
#define F_ZERO (0.0f, 0.0f, 0.0f, 0.0f) |
||||
float4 cvt(const uchar4 val) |
||||
{ |
||||
float4 result; |
||||
result.x = val.x; |
||||
result.y = val.y; |
||||
result.z = val.z; |
||||
result.w = val.w; |
||||
|
||||
return result; |
||||
} |
||||
|
||||
float sqr(const float4 val) |
||||
{ |
||||
return val.x * val.x + val.y * val.y + val.z * val.z; |
||||
} |
||||
|
||||
float sum(const float4 val) |
||||
{ |
||||
return (val.x + val.y + val.z); |
||||
} |
||||
|
||||
float4 clamp1(const float4 var, float learningRate, const float4 diff, float minVar) |
||||
{ |
||||
float4 result; |
||||
result.x = fmax(var.x + learningRate * (diff.x * diff.x - var.x), minVar); |
||||
result.y = fmax(var.y + learningRate * (diff.y * diff.y - var.y), minVar); |
||||
result.z = fmax(var.z + learningRate * (diff.z * diff.z - var.z), minVar); |
||||
result.w = 0.0f; |
||||
return result; |
||||
} |
||||
#endif |
||||
|
||||
typedef struct |
||||
{ |
||||
float c_Tb; |
||||
float c_TB; |
||||
float c_Tg; |
||||
float c_varInit; |
||||
float c_varMin; |
||||
float c_varMax; |
||||
float c_tau; |
||||
uchar c_shadowVal; |
||||
}con_srtuct_t; |
||||
|
||||
void swap(__global float* ptr, int x, int y, int k, int rows, int ptr_step) |
||||
{ |
||||
float val = ptr[(k * rows + y) * ptr_step + x]; |
||||
ptr[(k * rows + y) * ptr_step + x] = ptr[((k + 1) * rows + y) * ptr_step + x]; |
||||
ptr[((k + 1) * rows + y) * ptr_step + x] = val; |
||||
} |
||||
|
||||
void swap4(__global float4* ptr, int x, int y, int k, int rows, int ptr_step) |
||||
{ |
||||
float4 val = ptr[(k * rows + y) * ptr_step + x]; |
||||
ptr[(k * rows + y) * ptr_step + x] = ptr[((k + 1) * rows + y) * ptr_step + x]; |
||||
ptr[((k + 1) * rows + y) * ptr_step + x] = val; |
||||
} |
||||
|
||||
__kernel void mog_withoutLearning_kernel(__global T_FRAME* frame, __global uchar* fgmask, |
||||
__global float* weight, __global T_MEAN_VAR* mean, __global T_MEAN_VAR* var, |
||||
int frame_row, int frame_col, int frame_step, int fgmask_step, |
||||
int weight_step, int mean_step, int var_step, |
||||
float varThreshold, float backgroundRatio, int fgmask_offset_x, |
||||
int fgmask_offset_y, int frame_offset_x, int frame_offset_y) |
||||
{ |
||||
int x = get_global_id(0); |
||||
int y = get_global_id(1); |
||||
|
||||
if (x < frame_col && y < frame_row) |
||||
{ |
||||
T_MEAN_VAR pix = cvt(frame[(y + frame_offset_y) * frame_step + (x + frame_offset_x)]); |
||||
|
||||
int kHit = -1; |
||||
int kForeground = -1; |
||||
|
||||
for (int k = 0; k < (NMIXTURES); ++k) |
||||
{ |
||||
if (weight[(k * frame_row + y) * weight_step + x] < 1.192092896e-07f) |
||||
break; |
||||
|
||||
T_MEAN_VAR mu = mean[(k * frame_row + y) * mean_step + x]; |
||||
T_MEAN_VAR _var = var[(k * frame_row + y) + var_step + x]; |
||||
|
||||
T_MEAN_VAR diff = pix - mu; |
||||
|
||||
if (sqr(diff) < varThreshold * sum(_var)) |
||||
{ |
||||
kHit = k; |
||||
break; |
||||
} |
||||
} |
||||
|
||||
if (kHit >= 0) |
||||
{ |
||||
float wsum = 0.0f; |
||||
for (int k = 0; k < (NMIXTURES); ++k) |
||||
{ |
||||
wsum += weight[(k * frame_row + y) * weight_step + x]; |
||||
|
||||
if (wsum > backgroundRatio) |
||||
{ |
||||
kForeground = k + 1; |
||||
break; |
||||
} |
||||
} |
||||
} |
||||
if(kHit < 0 || kHit >= kForeground) |
||||
fgmask[(y + fgmask_offset_y) * fgmask_step + (x + fgmask_offset_x)] = (uchar) (-1); |
||||
else |
||||
fgmask[(y + fgmask_offset_y) * fgmask_step + (x + fgmask_offset_x)] = (uchar) (0); |
||||
} |
||||
} |
||||
|
||||
__kernel void mog_withLearning_kernel(__global T_FRAME* frame, __global int* fgmask, |
||||
__global float* weight, __global float* sortKey, __global T_MEAN_VAR* mean, |
||||
__global T_MEAN_VAR* var, int frame_row, int frame_col, int frame_step, int fgmask_step, |
||||
int weight_step, int sortKey_step, int mean_step, int var_step, |
||||
float varThreshold, float backgroundRatio, float learningRate, float minVar, |
||||
int fgmask_offset_x, int fgmask_offset_y, int frame_offset_x, int frame_offset_y) |
||||
{ |
||||
const float w0 = 0.05f; |
||||
const float sk0 = w0 / 30.0f; |
||||
const float var0 = 900.f; |
||||
|
||||
int x = get_global_id(0); |
||||
int y = get_global_id(1); |
||||
|
||||
if(x >= frame_col || y >= frame_row) return; |
||||
float wsum = 0.0f; |
||||
int kHit = -1; |
||||
int kForeground = -1; |
||||
int k = 0; |
||||
|
||||
T_MEAN_VAR pix = cvt(frame[(y + frame_offset_y) * frame_step + (x + frame_offset_x)]); |
||||
|
||||
for (; k < (NMIXTURES); ++k) |
||||
{ |
||||
float w = weight[(k * frame_row + y) * weight_step + x]; |
||||
wsum += w; |
||||
|
||||
if (w < 1.192092896e-07f) |
||||
break; |
||||
|
||||
T_MEAN_VAR mu = mean[(k * frame_row + y) * mean_step + x]; |
||||
T_MEAN_VAR _var = var[(k * frame_row + y) * var_step + x]; |
||||
|
||||
float sortKey_prev, weight_prev; |
||||
T_MEAN_VAR mean_prev, var_prev; |
||||
if (sqr(pix - mu) < varThreshold * sum(_var)) |
||||
{ |
||||
wsum -= w; |
||||
float dw = learningRate * (1.0f - w); |
||||
|
||||
_var = clamp1(_var, learningRate, pix - mu, minVar); |
||||
|
||||
sortKey_prev = w / sqr(sum(_var)); |
||||
sortKey[(k * frame_row + y) * sortKey_step + x] = sortKey_prev; |
||||
|
||||
weight_prev = w + dw; |
||||
weight[(k * frame_row + y) * weight_step + x] = weight_prev; |
||||
|
||||
mean_prev = mu + learningRate * (pix - mu); |
||||
mean[(k * frame_row + y) * mean_step + x] = mean_prev; |
||||
|
||||
var_prev = _var; |
||||
var[(k * frame_row + y) * var_step + x] = var_prev; |
||||
} |
||||
|
||||
int k1 = k - 1; |
||||
|
||||
if (k1 >= 0 && sqr(pix - mu) < varThreshold * sum(_var)) |
||||
{ |
||||
float sortKey_next = sortKey[(k1 * frame_row + y) * sortKey_step + x]; |
||||
float weight_next = weight[(k1 * frame_row + y) * weight_step + x]; |
||||
T_MEAN_VAR mean_next = mean[(k1 * frame_row + y) * mean_step + x]; |
||||
T_MEAN_VAR var_next = var[(k1 * frame_row + y) * var_step + x]; |
||||
|
||||
for (; sortKey_next < sortKey_prev && k1 >= 0; --k1) |
||||
{ |
||||
sortKey[(k1 * frame_row + y) * sortKey_step + x] = sortKey_prev; |
||||
sortKey[((k1 + 1) * frame_row + y) * sortKey_step + x] = sortKey_next; |
||||
|
||||
weight[(k1 * frame_row + y) * weight_step + x] = weight_prev; |
||||
weight[((k1 + 1) * frame_row + y) * weight_step + x] = weight_next; |
||||
|
||||
mean[(k1 * frame_row + y) * mean_step + x] = mean_prev; |
||||
mean[((k1 + 1) * frame_row + y) * mean_step + x] = mean_next; |
||||
|
||||
var[(k1 * frame_row + y) * var_step + x] = var_prev; |
||||
var[((k1 + 1) * frame_row + y) * var_step + x] = var_next; |
||||
|
||||
sortKey_prev = sortKey_next; |
||||
sortKey_next = k1 > 0 ? sortKey[((k1 - 1) * frame_row + y) * sortKey_step + x] : 0.0f; |
||||
|
||||
weight_prev = weight_next; |
||||
weight_next = k1 > 0 ? weight[((k1 - 1) * frame_row + y) * weight_step + x] : 0.0f; |
||||
|
||||
mean_prev = mean_next; |
||||
mean_next = k1 > 0 ? mean[((k1 - 1) * frame_row + y) * mean_step + x] : (T_MEAN_VAR)F_ZERO; |
||||
|
||||
var_prev = var_next; |
||||
var_next = k1 > 0 ? var[((k1 - 1) * frame_row + y) * var_step + x] : (T_MEAN_VAR)F_ZERO; |
||||
} |
||||
} |
||||
|
||||
kHit = k1 + 1; |
||||
break; |
||||
} |
||||
|
||||
if (kHit < 0) |
||||
{ |
||||
kHit = k = k < ((NMIXTURES) - 1) ? k : ((NMIXTURES) - 1); |
||||
wsum += w0 - weight[(k * frame_row + y) * weight_step + x]; |
||||
|
||||
weight[(k * frame_row + y) * weight_step + x] = w0; |
||||
mean[(k * frame_row + y) * mean_step + x] = pix; |
||||
#if defined (CN1) |
||||
var[(k * frame_row + y) * var_step + x] = (T_MEAN_VAR)(var0); |
||||
#else |
||||
var[(k * frame_row + y) * var_step + x] = (T_MEAN_VAR)(var0, var0, var0, var0); |
||||
#endif |
||||
sortKey[(k * frame_row + y) * sortKey_step + x] = sk0; |
||||
} |
||||
else |
||||
{ |
||||
for( ; k < (NMIXTURES); k++) |
||||
wsum += weight[(k * frame_row + y) * weight_step + x]; |
||||
} |
||||
|
||||
float wscale = 1.0f / wsum; |
||||
wsum = 0; |
||||
for (k = 0; k < (NMIXTURES); ++k) |
||||
{ |
||||
float w = weight[(k * frame_row + y) * weight_step + x]; |
||||
w *= wscale; |
||||
wsum += w; |
||||
|
||||
weight[(k * frame_row + y) * weight_step + x] = w; |
||||
sortKey[(k * frame_row + y) * sortKey_step + x] *= wscale; |
||||
|
||||
kForeground = select(kForeground, k + 1, wsum > backgroundRatio && kForeground < 0); |
||||
} |
||||
fgmask[(y + fgmask_offset_y) * fgmask_step + (x + fgmask_offset_x)] = (uchar)(-(kHit >= kForeground)); |
||||
} |
||||
|
||||
|
||||
__kernel void getBackgroundImage_kernel(__global float* weight, __global T_MEAN_VAR* mean, __global T_FRAME* dst, |
||||
int dst_row, int dst_col, int weight_step, int mean_step, int dst_step, |
||||
float backgroundRatio) |
||||
{ |
||||
int x = get_global_id(0); |
||||
int y = get_global_id(1); |
||||
|
||||
if(x < dst_col && y < dst_row) |
||||
{ |
||||
T_MEAN_VAR meanVal = (T_MEAN_VAR)F_ZERO; |
||||
float totalWeight = 0.0f; |
||||
|
||||
for (int mode = 0; mode < (NMIXTURES); ++mode) |
||||
{ |
||||
float _weight = weight[(mode * dst_row + y) * weight_step + x]; |
||||
|
||||
T_MEAN_VAR _mean = mean[(mode * dst_row + y) * mean_step + x]; |
||||
meanVal = meanVal + _weight * _mean; |
||||
|
||||
totalWeight += _weight; |
||||
|
||||
if(totalWeight > backgroundRatio) |
||||
break; |
||||
} |
||||
meanVal = meanVal * (1.f / totalWeight); |
||||
dst[y * dst_step + x] = CONVERT_TYPE(meanVal); |
||||
} |
||||
} |
||||
|
||||
__kernel void mog2_kernel(__global T_FRAME * frame, __global int* fgmask, __global float* weight, __global T_MEAN_VAR * mean, |
||||
__global int* modesUsed, __global float* variance, int frame_row, int frame_col, int frame_step, |
||||
int fgmask_step, int weight_step, int mean_step, int modesUsed_step, int var_step, float alphaT, float alpha1, float prune, |
||||
int detectShadows_flag, int fgmask_offset_x, int fgmask_offset_y, int frame_offset_x, int frame_offset_y, __constant con_srtuct_t* constants) |
||||
{ |
||||
int x = get_global_id(0); |
||||
int y = get_global_id(1); |
||||
|
||||
if(x < frame_col && y < frame_row) |
||||
{ |
||||
T_MEAN_VAR pix = cvt(frame[(y + frame_offset_y) * frame_step + x + frame_offset_x]); |
||||
|
||||
bool background = false; // true - the pixel classified as background |
||||
|
||||
bool fitsPDF = false; //if it remains zero a new GMM mode will be added |
||||
|
||||
int nmodes = modesUsed[y * modesUsed_step + x]; |
||||
int nNewModes = nmodes; //current number of modes in GMM |
||||
|
||||
float totalWeight = 0.0f; |
||||
|
||||
for (int mode = 0; mode < nmodes; ++mode) |
||||
{ |
||||
float _weight = alpha1 * weight[(mode * frame_row + y) * weight_step + x] + prune; |
||||
|
||||
if (!fitsPDF) |
||||
{ |
||||
float var = variance[(mode * frame_row + y) * var_step + x]; |
||||
|
||||
T_MEAN_VAR _mean = mean[(mode * frame_row + y) * mean_step + x]; |
||||
|
||||
T_MEAN_VAR diff = _mean - pix; |
||||
float dist2 = sqr(diff); |
||||
|
||||
if (totalWeight < constants -> c_TB && dist2 < constants -> c_Tb * var) |
||||
background = true; |
||||
|
||||
if (dist2 < constants -> c_Tg * var) |
||||
{ |
||||
fitsPDF = true; |
||||
_weight += alphaT; |
||||
float k = alphaT / _weight; |
||||
mean[(mode * frame_row + y) * mean_step + x] = _mean - k * diff; |
||||
float varnew = var + k * (dist2 - var); |
||||
varnew = fmax(varnew, constants -> c_varMin); |
||||
varnew = fmin(varnew, constants -> c_varMax); |
||||
|
||||
variance[(mode * frame_row + y) * var_step + x] = varnew; |
||||
for (int i = mode; i > 0; --i) |
||||
{ |
||||
if (_weight < weight[((i - 1) * frame_row + y) * weight_step + x]) |
||||
break; |
||||
swap(weight, x, y, i - 1, frame_row, weight_step); |
||||
swap(variance, x, y, i - 1, frame_row, var_step); |
||||
#if defined (CN1) |
||||
swap(mean, x, y, i - 1, frame_row, mean_step); |
||||
#else |
||||
swap4(mean, x, y, i - 1, frame_row, mean_step); |
||||
#endif |
||||
} |
||||
} |
||||
} // !fitsPDF |
||||
|
||||
if (_weight < -prune) |
||||
{ |
||||
_weight = 0.0; |
||||
nmodes--; |
||||
} |
||||
|
||||
weight[(mode * frame_row + y) * weight_step + x] = _weight; //update weight by the calculated value |
||||
totalWeight += _weight; |
||||
} |
||||
|
||||
totalWeight = 1.f / totalWeight; |
||||
for (int mode = 0; mode < nmodes; ++mode) |
||||
weight[(mode * frame_row + y) * weight_step + x] *= totalWeight; |
||||
|
||||
nmodes = nNewModes; |
||||
|
||||
if (!fitsPDF) |
||||
{ |
||||
int mode = nmodes == (NMIXTURES) ? (NMIXTURES) - 1 : nmodes++; |
||||
|
||||
if (nmodes == 1) |
||||
weight[(mode * frame_row + y) * weight_step + x] = 1.f; |
||||
else |
||||
{ |
||||
weight[(mode * frame_row + y) * weight_step + x] = alphaT; |
||||
|
||||
for (int i = 0; i < nmodes - 1; ++i) |
||||
weight[(i * frame_row + y) * weight_step + x] *= alpha1; |
||||
} |
||||
|
||||
mean[(mode * frame_row + y) * mean_step + x] = pix; |
||||
variance[(mode * frame_row + y) * var_step + x] = constants -> c_varInit; |
||||
|
||||
for (int i = nmodes - 1; i > 0; --i) |
||||
{ |
||||
// check one up |
||||
if (alphaT < weight[((i - 1) * frame_row + y) * weight_step + x]) |
||||
break; |
||||
|
||||
swap(weight, x, y, i - 1, frame_row, weight_step); |
||||
swap(variance, x, y, i - 1, frame_row, var_step); |
||||
#if defined (CN1) |
||||
swap(mean, x, y, i - 1, frame_row, mean_step); |
||||
#else |
||||
swap4(mean, x, y, i - 1, frame_row, mean_step); |
||||
#endif |
||||
} |
||||
} |
||||
|
||||
modesUsed[y * modesUsed_step + x] = nmodes; |
||||
|
||||
bool isShadow = false; |
||||
if (detectShadows_flag && !background) |
||||
{ |
||||
float tWeight = 0.0f; |
||||
|
||||
for (int mode = 0; mode < nmodes; ++mode) |
||||
{ |
||||
T_MEAN_VAR _mean = mean[(mode * frame_row + y) * mean_step + x]; |
||||
|
||||
T_MEAN_VAR pix_mean = pix * _mean; |
||||
|
||||
float numerator = sum(pix_mean); |
||||
float denominator = sqr(_mean); |
||||
|
||||
if (denominator == 0) |
||||
break; |
||||
|
||||
if (numerator <= denominator && numerator >= constants -> c_tau * denominator) |
||||
{ |
||||
float a = numerator / denominator; |
||||
|
||||
T_MEAN_VAR dD = a * _mean - pix; |
||||
|
||||
if (sqr(dD) < constants -> c_Tb * variance[(mode * frame_row + y) * var_step + x] * a * a) |
||||
{ |
||||
isShadow = true; |
||||
break; |
||||
} |
||||
} |
||||
|
||||
tWeight += weight[(mode * frame_row + y) * weight_step + x]; |
||||
if (tWeight > constants -> c_TB) |
||||
break; |
||||
} |
||||
} |
||||
|
||||
fgmask[(y + fgmask_offset_y) * fgmask_step + x + fgmask_offset_x] = background ? 0 : isShadow ? constants -> c_shadowVal : 255; |
||||
} |
||||
} |
||||
|
||||
__kernel void getBackgroundImage2_kernel(__global int* modesUsed, __global float* weight, __global T_MEAN_VAR* mean, |
||||
__global T_FRAME* dst, float c_TB, int modesUsed_row, int modesUsed_col, int modesUsed_step, int weight_step, |
||||
int mean_step, int dst_step, int dst_x, int dst_y) |
||||
{ |
||||
int x = get_global_id(0); |
||||
int y = get_global_id(1); |
||||
|
||||
if(x < modesUsed_col && y < modesUsed_row) |
||||
{ |
||||
int nmodes = modesUsed[y * modesUsed_step + x]; |
||||
|
||||
T_MEAN_VAR meanVal = (T_MEAN_VAR)F_ZERO; |
||||
|
||||
float totalWeight = 0.0f; |
||||
|
||||
for (int mode = 0; mode < nmodes; ++mode) |
||||
{ |
||||
float _weight = weight[(mode * modesUsed_row + y) * weight_step + x]; |
||||
|
||||
T_MEAN_VAR _mean = mean[(mode * modesUsed_row + y) * mean_step + x]; |
||||
meanVal = meanVal + _weight * _mean; |
||||
|
||||
totalWeight += _weight; |
||||
|
||||
if(totalWeight > c_TB) |
||||
break; |
||||
} |
||||
|
||||
meanVal = meanVal * (1.f / totalWeight); |
||||
dst[(y + dst_y) * dst_step + x + dst_x] = CONVERT_TYPE(meanVal); |
||||
} |
||||
} |
@ -0,0 +1,424 @@ |
||||
/*M/////////////////////////////////////////////////////////////////////////////////////// |
||||
// |
||||
// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING. |
||||
// |
||||
// By downloading, copying, installing or using the software you agree to this license. |
||||
// If you do not agree to this license, do not download, install, |
||||
// copy or use the software. |
||||
// |
||||
// |
||||
// License Agreement |
||||
// For Open Source Computer Vision Library |
||||
// |
||||
// Copyright (C) 2010-2013, Multicoreware, Inc., all rights reserved. |
||||
// Copyright (C) 2010-2013, Advanced Micro Devices, Inc., all rights reserved. |
||||
// Third party copyrights are property of their respective owners. |
||||
// |
||||
// @Authors |
||||
// Harris Gasparakis, harris.gasparakis@amd.com |
||||
// Xiaopeng Fu, fuxiaopeng2222@163.com |
||||
// Yao Wang, bitwangyaoyao@gmail.com |
||||
// |
||||
// Redistribution and use in source and binary forms, with or without modification, |
||||
// are permitted provided that the following conditions are met: |
||||
// |
||||
// * Redistribution's of source code must retain the above copyright notice, |
||||
// this list of conditions and the following disclaimer. |
||||
// |
||||
// * Redistribution's in binary form must reproduce the above copyright notice, |
||||
// this list of conditions and the following disclaimer in the documentation |
||||
// and/or other oclMaterials provided with the distribution. |
||||
// |
||||
// * The name of the copyright holders may not be used to endorse or promote products |
||||
// derived from this software without specific prior written permission. |
||||
// |
||||
// This software is provided by the copyright holders and contributors as is and |
||||
// any express or implied warranties, including, but not limited to, the implied |
||||
// warranties of merchantability and fitness for a particular purpose are disclaimed. |
||||
// In no event shall the Intel Corporation or contributors be liable for any direct, |
||||
// indirect, incidental, special, exemplary, or consequential damages |
||||
// (including, but not limited to, procurement of substitute goods or services; |
||||
// loss of use, data, or profits; or business interruption) however caused |
||||
// and on any theory of liability, whether in contract, strict liability, |
||||
// or tort (including negligence or otherwise) arising in any way out of |
||||
// the use of this software, even if advised of the possibility of such damage. |
||||
// |
||||
//M*/ |
||||
|
||||
|
||||
#ifdef BORDER_REPLICATE |
||||
//BORDER_REPLICATE: aaaaaa|abcdefgh|hhhhhhh |
||||
#define ADDR_L(i, l_edge, r_edge) ((i) < (l_edge) ? (l_edge) : (i)) |
||||
#define ADDR_R(i, r_edge, addr) ((i) >= (r_edge) ? (r_edge)-1 : (addr)) |
||||
#define ADDR_H(i, t_edge, b_edge) ((i) < (t_edge) ? (t_edge) :(i)) |
||||
#define ADDR_B(i, b_edge, addr) ((i) >= (b_edge) ? (b_edge)-1 :(addr)) |
||||
#endif |
||||
|
||||
#ifdef BORDER_REFLECT |
||||
//BORDER_REFLECT: fedcba|abcdefgh|hgfedcb |
||||
#define ADDR_L(i, l_edge, r_edge) ((i) < (l_edge) ? -(i)-1 : (i)) |
||||
#define ADDR_R(i, r_edge, addr) ((i) >= (r_edge) ? -(i)-1+((r_edge)<<1) : (addr)) |
||||
#define ADDR_H(i, t_edge, b_edge) ((i) < (t_edge) ? -(i)-1 : (i)) |
||||
#define ADDR_B(i, b_edge, addr) ((i) >= (b_edge) ? -(i)-1+((b_edge)<<1) : (addr)) |
||||
#endif |
||||
|
||||
#ifdef BORDER_REFLECT_101 |
||||
//BORDER_REFLECT_101: gfedcb|abcdefgh|gfedcba |
||||
#define ADDR_L(i, l_edge, r_edge) ((i) < (l_edge) ? -(i) : (i)) |
||||
#define ADDR_R(i, r_edge, addr) ((i) >= (r_edge) ? -(i)-2+((r_edge)<<1) : (addr)) |
||||
#define ADDR_H(i, t_edge, b_edge) ((i) < (t_edge) ? -(i) : (i)) |
||||
#define ADDR_B(i, b_edge, addr) ((i) >= (b_edge) ? -(i)-2+((b_edge)<<1) : (addr)) |
||||
#endif |
||||
|
||||
//blur function does not support BORDER_WRAP |
||||
#ifdef BORDER_WRAP |
||||
//BORDER_WRAP: cdefgh|abcdefgh|abcdefg |
||||
#define ADDR_L(i, l_edge, r_edge) ((i) < (l_edge) ? (i)+(r_edge) : (i)) |
||||
#define ADDR_R(i, r_edge, addr) ((i) >= (r_edge) ? (i)-(r_edge) : (addr)) |
||||
#define ADDR_H(i, t_edge, b_edge) ((i) < (t_edge) ? (i)+(b_edge) : (i)) |
||||
#define ADDR_B(i, b_edge, addr) ((i) >= (b_edge) ? (i)-(b_edge) : (addr)) |
||||
#endif |
||||
|
||||
__kernel void |
||||
edgeEnhancingFilter_C4_D0( |
||||
__global const uchar4 * restrict src, |
||||
__global uchar4 *dst, |
||||
float alpha, |
||||
int src_offset, |
||||
int src_whole_rows, |
||||
int src_whole_cols, |
||||
int src_step, |
||||
int dst_offset, |
||||
int dst_rows, |
||||
int dst_cols, |
||||
int dst_step, |
||||
__global const float* lut, |
||||
int lut_step) |
||||
{ |
||||
int col = get_local_id(0); |
||||
const int gX = get_group_id(0); |
||||
const int gY = get_group_id(1); |
||||
|
||||
int src_x_off = (src_offset % src_step) >> 2; |
||||
int src_y_off = src_offset / src_step; |
||||
int dst_x_off = (dst_offset % dst_step) >> 2; |
||||
int dst_y_off = dst_offset / dst_step; |
||||
|
||||
int startX = gX * (THREADS-ksX+1) - anX + src_x_off; |
||||
int startY = (gY * (1+EXTRA)) - anY + src_y_off; |
||||
|
||||
int dst_startX = gX * (THREADS-ksX+1) + dst_x_off; |
||||
int dst_startY = (gY * (1+EXTRA)) + dst_y_off; |
||||
|
||||
int posX = dst_startX - dst_x_off + col; |
||||
int posY = (gY * (1+EXTRA)) ; |
||||
|
||||
__local uchar4 data[ksY+EXTRA][THREADS]; |
||||
|
||||
float4 tmp_sum[1+EXTRA]; |
||||
for(int tmpint = 0; tmpint < 1+EXTRA; tmpint++) |
||||
{ |
||||
tmp_sum[tmpint] = (float4)(0,0,0,0); |
||||
} |
||||
|
||||
#ifdef BORDER_CONSTANT |
||||
bool con; |
||||
uchar4 ss; |
||||
for(int j = 0; j < ksY+EXTRA; j++) |
||||
{ |
||||
con = (startX+col >= 0 && startX+col < src_whole_cols && startY+j >= 0 && startY+j < src_whole_rows); |
||||
|
||||
int cur_col = clamp(startX + col, 0, src_whole_cols); |
||||
if(con) |
||||
{ |
||||
ss = src[(startY+j)*(src_step>>2) + cur_col]; |
||||
} |
||||
|
||||
data[j][col] = con ? ss : (uchar4)0; |
||||
} |
||||
#else |
||||
for(int j= 0; j < ksY+EXTRA; j++) |
||||
{ |
||||
int selected_row; |
||||
int selected_col; |
||||
selected_row = ADDR_H(startY+j, 0, src_whole_rows); |
||||
selected_row = ADDR_B(startY+j, src_whole_rows, selected_row); |
||||
|
||||
selected_col = ADDR_L(startX+col, 0, src_whole_cols); |
||||
selected_col = ADDR_R(startX+col, src_whole_cols, selected_col); |
||||
|
||||
data[j][col] = src[selected_row * (src_step>>2) + selected_col]; |
||||
} |
||||
#endif |
||||
|
||||
barrier(CLK_LOCAL_MEM_FENCE); |
||||
|
||||
float4 var[1+EXTRA]; |
||||
|
||||
#if VAR_PER_CHANNEL |
||||
float4 weight; |
||||
float4 totalWeight = (float4)(0,0,0,0); |
||||
#else |
||||
float weight; |
||||
float totalWeight = 0; |
||||
#endif |
||||
|
||||
int4 currValCenter; |
||||
int4 currWRTCenter; |
||||
|
||||
int4 sumVal = 0; |
||||
int4 sumValSqr = 0; |
||||
|
||||
if(col < (THREADS-(ksX-1))) |
||||
{ |
||||
int4 currVal; |
||||
|
||||
int howManyAll = (2*anX+1)*(ksY); |
||||
|
||||
//find variance of all data |
||||
int startLMj; |
||||
int endLMj ; |
||||
#if CALCVAR |
||||
// Top row: don't sum the very last element |
||||
for(int extraCnt = 0; extraCnt <=EXTRA; extraCnt++) |
||||
{ |
||||
startLMj = extraCnt; |
||||
endLMj = ksY+extraCnt-1; |
||||
sumVal =0; |
||||
sumValSqr=0; |
||||
for(int j = startLMj; j < endLMj; j++) |
||||
{ |
||||
for(int i=-anX; i<=anX; i++) |
||||
{ |
||||
currVal = convert_int4(data[j][col+anX+i]) ; |
||||
|
||||
sumVal += currVal; |
||||
sumValSqr += mul24(currVal, currVal); |
||||
} |
||||
} |
||||
var[extraCnt] = convert_float4( ( (sumValSqr * howManyAll)- mul24(sumVal , sumVal) ) ) / ( (float)(howManyAll*howManyAll) ) ; |
||||
#else |
||||
var[extraCnt] = (float4)(900.0, 900.0, 900.0, 0.0); |
||||
#endif |
||||
} |
||||
|
||||
for(int extraCnt = 0; extraCnt <= EXTRA; extraCnt++) |
||||
{ |
||||
|
||||
// top row: include the very first element, even on first time |
||||
startLMj = extraCnt; |
||||
// go all the way, unless this is the last local mem chunk, |
||||
// then stay within limits - 1 |
||||
endLMj = extraCnt + ksY; |
||||
|
||||
// Top row: don't sum the very last element |
||||
currValCenter = convert_int4( data[ (startLMj + endLMj)/2][col+anX] ); |
||||
|
||||
for(int j = startLMj, lut_j = 0; j < endLMj; j++, lut_j++) |
||||
{ |
||||
for(int i=-anX; i<=anX; i++) |
||||
{ |
||||
#if FIXED_WEIGHT |
||||
#if VAR_PER_CHANNEL |
||||
weight.x = 1.0f; |
||||
weight.y = 1.0f; |
||||
weight.z = 1.0f; |
||||
weight.w = 1.0f; |
||||
#else |
||||
weight = 1.0f; |
||||
#endif |
||||
#else |
||||
currVal = convert_int4(data[j][col+anX+i]) ; |
||||
currWRTCenter = currVal-currValCenter; |
||||
|
||||
#if VAR_PER_CHANNEL |
||||
weight = var[extraCnt] / (var[extraCnt] + convert_float4(currWRTCenter * currWRTCenter)) * (float4)(lut[lut_j*lut_step+anX+i]); |
||||
//weight.x = var[extraCnt].x / ( var[extraCnt].x + (float) mul24(currWRTCenter.x , currWRTCenter.x) ) ; |
||||
//weight.y = var[extraCnt].y / ( var[extraCnt].y + (float) mul24(currWRTCenter.y , currWRTCenter.y) ) ; |
||||
//weight.z = var[extraCnt].z / ( var[extraCnt].z + (float) mul24(currWRTCenter.z , currWRTCenter.z) ) ; |
||||
//weight.w = 0; |
||||
#else |
||||
weight = 1.0f/(1.0f+( mul24(currWRTCenter.x, currWRTCenter.x) + mul24(currWRTCenter.y, currWRTCenter.y) + mul24(currWRTCenter.z, currWRTCenter.z))/(var.x+var.y+var.z)); |
||||
#endif |
||||
#endif |
||||
tmp_sum[extraCnt] += convert_float4(data[j][col+anX+i]) * weight; |
||||
totalWeight += weight; |
||||
} |
||||
} |
||||
|
||||
tmp_sum[extraCnt] /= totalWeight; |
||||
|
||||
if(posX >= 0 && posX < dst_cols && (posY+extraCnt) >= 0 && (posY+extraCnt) < dst_rows) |
||||
{ |
||||
dst[(dst_startY+extraCnt) * (dst_step>>2)+ dst_startX + col] = convert_uchar4(tmp_sum[extraCnt]); |
||||
} |
||||
|
||||
#if VAR_PER_CHANNEL |
||||
totalWeight = (float4)(0,0,0,0); |
||||
#else |
||||
totalWeight = 0; |
||||
#endif |
||||
} |
||||
} |
||||
} |
||||
|
||||
|
||||
__kernel void |
||||
edgeEnhancingFilter_C1_D0( |
||||
__global const uchar * restrict src, |
||||
__global uchar *dst, |
||||
float alpha, |
||||
int src_offset, |
||||
int src_whole_rows, |
||||
int src_whole_cols, |
||||
int src_step, |
||||
int dst_offset, |
||||
int dst_rows, |
||||
int dst_cols, |
||||
int dst_step, |
||||
__global const float * lut, |
||||
int lut_step) |
||||
{ |
||||
int col = get_local_id(0); |
||||
const int gX = get_group_id(0); |
||||
const int gY = get_group_id(1); |
||||
|
||||
int src_x_off = (src_offset % src_step); |
||||
int src_y_off = src_offset / src_step; |
||||
int dst_x_off = (dst_offset % dst_step); |
||||
int dst_y_off = dst_offset / dst_step; |
||||
|
||||
int startX = gX * (THREADS-ksX+1) - anX + src_x_off; |
||||
int startY = (gY * (1+EXTRA)) - anY + src_y_off; |
||||
|
||||
int dst_startX = gX * (THREADS-ksX+1) + dst_x_off; |
||||
int dst_startY = (gY * (1+EXTRA)) + dst_y_off; |
||||
|
||||
int posX = dst_startX - dst_x_off + col; |
||||
int posY = (gY * (1+EXTRA)) ; |
||||
|
||||
__local uchar data[ksY+EXTRA][THREADS]; |
||||
|
||||
float tmp_sum[1+EXTRA]; |
||||
for(int tmpint = 0; tmpint < 1+EXTRA; tmpint++) |
||||
{ |
||||
tmp_sum[tmpint] = (float)(0); |
||||
} |
||||
|
||||
#ifdef BORDER_CONSTANT |
||||
bool con; |
||||
uchar ss; |
||||
for(int j = 0; j < ksY+EXTRA; j++) |
||||
{ |
||||
con = (startX+col >= 0 && startX+col < src_whole_cols && startY+j >= 0 && startY+j < src_whole_rows); |
||||
|
||||
int cur_col = clamp(startX + col, 0, src_whole_cols); |
||||
if(con) |
||||
{ |
||||
ss = src[(startY+j)*(src_step) + cur_col]; |
||||
} |
||||
|
||||
data[j][col] = con ? ss : 0; |
||||
} |
||||
#else |
||||
for(int j= 0; j < ksY+EXTRA; j++) |
||||
{ |
||||
int selected_row; |
||||
int selected_col; |
||||
selected_row = ADDR_H(startY+j, 0, src_whole_rows); |
||||
selected_row = ADDR_B(startY+j, src_whole_rows, selected_row); |
||||
|
||||
selected_col = ADDR_L(startX+col, 0, src_whole_cols); |
||||
selected_col = ADDR_R(startX+col, src_whole_cols, selected_col); |
||||
|
||||
data[j][col] = src[selected_row * (src_step) + selected_col]; |
||||
} |
||||
#endif |
||||
|
||||
barrier(CLK_LOCAL_MEM_FENCE); |
||||
|
||||
float var[1+EXTRA]; |
||||
|
||||
float weight; |
||||
float totalWeight = 0; |
||||
|
||||
int currValCenter; |
||||
int currWRTCenter; |
||||
|
||||
int sumVal = 0; |
||||
int sumValSqr = 0; |
||||
|
||||
if(col < (THREADS-(ksX-1))) |
||||
{ |
||||
int currVal; |
||||
|
||||
int howManyAll = (2*anX+1)*(ksY); |
||||
|
||||
//find variance of all data |
||||
int startLMj; |
||||
int endLMj; |
||||
#if CALCVAR |
||||
// Top row: don't sum the very last element |
||||
for(int extraCnt=0; extraCnt<=EXTRA; extraCnt++) |
||||
{ |
||||
startLMj = extraCnt; |
||||
endLMj = ksY+extraCnt-1; |
||||
sumVal = 0; |
||||
sumValSqr =0; |
||||
for(int j = startLMj; j < endLMj; j++) |
||||
{ |
||||
for(int i=-anX; i<=anX; i++) |
||||
{ |
||||
currVal = (uint)(data[j][col+anX+i]) ; |
||||
|
||||
sumVal += currVal; |
||||
sumValSqr += mul24(currVal, currVal); |
||||
} |
||||
} |
||||
var[extraCnt] = (float)( ( (sumValSqr * howManyAll)- mul24(sumVal , sumVal) ) ) / ( (float)(howManyAll*howManyAll) ) ; |
||||
#else |
||||
var[extraCnt] = (float)(900.0); |
||||
#endif |
||||
} |
||||
|
||||
for(int extraCnt = 0; extraCnt <= EXTRA; extraCnt++) |
||||
{ |
||||
|
||||
// top row: include the very first element, even on first time |
||||
startLMj = extraCnt; |
||||
// go all the way, unless this is the last local mem chunk, |
||||
// then stay within limits - 1 |
||||
endLMj = extraCnt + ksY; |
||||
|
||||
// Top row: don't sum the very last element |
||||
currValCenter = (int)( data[ (startLMj + endLMj)/2][col+anX] ); |
||||
|
||||
for(int j = startLMj, lut_j = 0; j < endLMj; j++, lut_j++) |
||||
{ |
||||
for(int i=-anX; i<=anX; i++) |
||||
{ |
||||
#if FIXED_WEIGHT |
||||
weight = 1.0f; |
||||
#else |
||||
currVal = (int)(data[j][col+anX+i]) ; |
||||
currWRTCenter = currVal-currValCenter; |
||||
|
||||
weight = var[extraCnt] / (var[extraCnt] + (float)mul24(currWRTCenter,currWRTCenter)) * lut[lut_j*lut_step+anX+i] ; |
||||
#endif |
||||
tmp_sum[extraCnt] += (float)(data[j][col+anX+i] * weight); |
||||
totalWeight += weight; |
||||
} |
||||
} |
||||
|
||||
tmp_sum[extraCnt] /= totalWeight; |
||||
|
||||
|
||||
if(posX >= 0 && posX < dst_cols && (posY+extraCnt) >= 0 && (posY+extraCnt) < dst_rows) |
||||
{ |
||||
dst[(dst_startY+extraCnt) * (dst_step)+ dst_startX + col] = (uchar)(tmp_sum[extraCnt]); |
||||
} |
||||
|
||||
totalWeight = 0; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,227 @@ |
||||
/*M///////////////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.
|
||||
//
|
||||
// By downloading, copying, installing or using the software you agree to this license.
|
||||
// If you do not agree to this license, do not download, install,
|
||||
// copy or use the software.
|
||||
//
|
||||
//
|
||||
// License Agreement
|
||||
// For Open Source Computer Vision Library
|
||||
//
|
||||
// Copyright (C) 2010-2013, Multicoreware, Inc., all rights reserved.
|
||||
// Copyright (C) 2010-2013, Advanced Micro Devices, Inc., all rights reserved.
|
||||
// Third party copyrights are property of their respective owners.
|
||||
//
|
||||
// @Authors
|
||||
// Jin Ma, jin@multicorewareinc.com
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without modification,
|
||||
// are permitted provided that the following conditions are met:
|
||||
//
|
||||
// * Redistribution's of source code must retain the above copyright notice,
|
||||
// this list of conditions and the following disclaimer.
|
||||
//
|
||||
// * Redistribution's in binary form must reproduce the above copyright notice,
|
||||
// this list of conditions and the following disclaimer in the documentation
|
||||
// and/or other oclMaterials provided with the distribution.
|
||||
//
|
||||
// * The name of the copyright holders may not be used to endorse or promote products
|
||||
// derived from this software without specific prior written permission.
|
||||
//
|
||||
// This software is provided by the copyright holders and contributors "as is" and
|
||||
// any express or implied warranties, including, but not limited to, the implied
|
||||
// warranties of merchantability and fitness for a particular purpose are disclaimed.
|
||||
// In no event shall the Intel Corporation or contributors be liable for any direct,
|
||||
// indirect, incidental, special, exemplary, or consequential damages
|
||||
// (including, but not limited to, procurement of substitute goods or services;
|
||||
// loss of use, data, or profits; or business interruption) however caused
|
||||
// and on any theory of liability, whether in contract, strict liability,
|
||||
// or tort (including negligence or otherwise) arising in any way out of
|
||||
// the use of this software, even if advised of the possibility of such damage.
|
||||
//
|
||||
//M*/
|
||||
|
||||
#include "test_precomp.hpp" |
||||
|
||||
#ifdef HAVE_OPENCL |
||||
|
||||
using namespace cv; |
||||
using namespace cv::ocl; |
||||
using namespace cvtest; |
||||
using namespace testing; |
||||
using namespace std; |
||||
|
||||
extern string workdir; |
||||
//////////////////////////////////////////////////////
|
||||
// MOG
|
||||
|
||||
namespace |
||||
{ |
||||
IMPLEMENT_PARAM_CLASS(UseGray, bool) |
||||
IMPLEMENT_PARAM_CLASS(LearningRate, double) |
||||
} |
||||
|
||||
PARAM_TEST_CASE(mog, UseGray, LearningRate, bool) |
||||
{ |
||||
bool useGray; |
||||
double learningRate; |
||||
bool useRoi; |
||||
|
||||
virtual void SetUp() |
||||
{ |
||||
useGray = GET_PARAM(0); |
||||
|
||||
learningRate = GET_PARAM(1); |
||||
|
||||
useRoi = GET_PARAM(2); |
||||
} |
||||
}; |
||||
|
||||
TEST_P(mog, Update) |
||||
{ |
||||
std::string inputFile = string(cvtest::TS::ptr()->get_data_path()) + "gpu/video/768x576.avi"; |
||||
cv::VideoCapture cap(inputFile); |
||||
ASSERT_TRUE(cap.isOpened()); |
||||
|
||||
cv::Mat frame; |
||||
cap >> frame; |
||||
ASSERT_FALSE(frame.empty()); |
||||
|
||||
cv::ocl::MOG mog; |
||||
cv::ocl::oclMat foreground = createMat_ocl(frame.size(), CV_8UC1, useRoi); |
||||
|
||||
Ptr<cv::BackgroundSubtractorMOG> mog_gold = createBackgroundSubtractorMOG(); |
||||
cv::Mat foreground_gold; |
||||
|
||||
for (int i = 0; i < 10; ++i) |
||||
{ |
||||
cap >> frame; |
||||
ASSERT_FALSE(frame.empty()); |
||||
|
||||
if (useGray) |
||||
{ |
||||
cv::Mat temp; |
||||
cv::cvtColor(frame, temp, cv::COLOR_BGR2GRAY); |
||||
cv::swap(temp, frame); |
||||
} |
||||
|
||||
mog(loadMat_ocl(frame, useRoi), foreground, (float)learningRate); |
||||
|
||||
mog_gold->apply(frame, foreground_gold, learningRate); |
||||
|
||||
EXPECT_MAT_NEAR(foreground_gold, foreground, 0.0); |
||||
} |
||||
} |
||||
INSTANTIATE_TEST_CASE_P(OCL_Video, mog, testing::Combine( |
||||
testing::Values(UseGray(false), UseGray(true)), |
||||
testing::Values(LearningRate(0.0), LearningRate(0.01)), |
||||
Values(true, false))); |
||||
|
||||
//////////////////////////////////////////////////////
|
||||
// MOG2
|
||||
|
||||
namespace |
||||
{ |
||||
IMPLEMENT_PARAM_CLASS(DetectShadow, bool) |
||||
} |
||||
|
||||
PARAM_TEST_CASE(mog2, UseGray, DetectShadow, bool) |
||||
{ |
||||
bool useGray; |
||||
bool detectShadow; |
||||
bool useRoi; |
||||
virtual void SetUp() |
||||
{ |
||||
useGray = GET_PARAM(0); |
||||
detectShadow = GET_PARAM(1); |
||||
useRoi = GET_PARAM(2); |
||||
} |
||||
}; |
||||
|
||||
TEST_P(mog2, Update) |
||||
{ |
||||
std::string inputFile = string(cvtest::TS::ptr()->get_data_path()) + "gpu/video/768x576.avi"; |
||||
cv::VideoCapture cap(inputFile); |
||||
ASSERT_TRUE(cap.isOpened()); |
||||
|
||||
cv::Mat frame; |
||||
cap >> frame; |
||||
ASSERT_FALSE(frame.empty()); |
||||
|
||||
cv::ocl::MOG2 mog2; |
||||
mog2.bShadowDetection = detectShadow; |
||||
cv::ocl::oclMat foreground = createMat_ocl(frame.size(), CV_8UC1, useRoi); |
||||
|
||||
cv::Ptr<cv::BackgroundSubtractorMOG2> mog2_gold = createBackgroundSubtractorMOG2(); |
||||
mog2_gold->set("detectShadows", detectShadow); |
||||
cv::Mat foreground_gold; |
||||
|
||||
for (int i = 0; i < 10; ++i) |
||||
{ |
||||
cap >> frame; |
||||
ASSERT_FALSE(frame.empty()); |
||||
|
||||
if (useGray) |
||||
{ |
||||
cv::Mat temp; |
||||
cv::cvtColor(frame, temp, cv::COLOR_BGR2GRAY); |
||||
cv::swap(temp, frame); |
||||
} |
||||
|
||||
mog2(loadMat_ocl(frame, useRoi), foreground); |
||||
|
||||
mog2_gold->apply(frame, foreground_gold); |
||||
|
||||
if (detectShadow) |
||||
EXPECT_MAT_SIMILAR(foreground_gold, foreground, 15e-3) |
||||
else |
||||
EXPECT_MAT_NEAR(foreground_gold, foreground, 0) |
||||
} |
||||
} |
||||
|
||||
TEST_P(mog2, getBackgroundImage) |
||||
{ |
||||
if (useGray) |
||||
return; |
||||
|
||||
std::string inputFile = string(cvtest::TS::ptr()->get_data_path()) + "gpu/video/768x576.avi"; |
||||
cv::VideoCapture cap(inputFile); |
||||
ASSERT_TRUE(cap.isOpened()); |
||||
|
||||
cv::Mat frame; |
||||
|
||||
cv::ocl::MOG2 mog2; |
||||
mog2.bShadowDetection = detectShadow; |
||||
cv::ocl::oclMat foreground; |
||||
|
||||
cv::Ptr<cv::BackgroundSubtractorMOG2> mog2_gold = createBackgroundSubtractorMOG2(); |
||||
mog2_gold->set("detectShadows", detectShadow); |
||||
cv::Mat foreground_gold; |
||||
|
||||
for (int i = 0; i < 10; ++i) |
||||
{ |
||||
cap >> frame; |
||||
ASSERT_FALSE(frame.empty()); |
||||
|
||||
mog2(loadMat_ocl(frame, useRoi), foreground); |
||||
|
||||
mog2_gold->apply(frame, foreground_gold); |
||||
} |
||||
|
||||
cv::ocl::oclMat background = createMat_ocl(frame.size(), frame.type(), useRoi); |
||||
mog2.getBackgroundImage(background); |
||||
|
||||
cv::Mat background_gold; |
||||
mog2_gold->getBackgroundImage(background_gold); |
||||
|
||||
EXPECT_MAT_NEAR(background_gold, background, 1.0); |
||||
} |
||||
|
||||
INSTANTIATE_TEST_CASE_P(OCL_Video, mog2, testing::Combine( |
||||
testing::Values(UseGray(true), UseGray(false)), |
||||
testing::Values(DetectShadow(true), DetectShadow(false)), |
||||
Values(true, false))); |
||||
|
||||
#endif |
@ -0,0 +1,52 @@ |
||||
// This sample shows the difference of adaptive bilateral filter and bilateral filter.
|
||||
#include "opencv2/core.hpp" |
||||
#include "opencv2/core/utility.hpp" |
||||
#include "opencv2/imgproc.hpp" |
||||
#include "opencv2/highgui.hpp" |
||||
#include "opencv2/ocl.hpp" |
||||
|
||||
using namespace cv; |
||||
using namespace std; |
||||
|
||||
|
||||
int main( int argc, const char** argv ) |
||||
{ |
||||
const char* keys = |
||||
"{ i input | | specify input image }" |
||||
"{ k ksize | 5 | specify kernel size }"; |
||||
CommandLineParser cmd(argc, argv, keys); |
||||
string src_path = cmd.get<string>("i"); |
||||
int ks = cmd.get<int>("k"); |
||||
const char * winName[] = {"input", "adaptive bilateral CPU", "adaptive bilateral OpenCL", "bilateralFilter OpenCL"}; |
||||
|
||||
Mat src = imread(src_path); |
||||
Mat abFilterCPU; |
||||
if(src.empty()){ |
||||
//cout << "error read image: " << src_path << endl;
|
||||
return -1; |
||||
} |
||||
|
||||
std::vector<ocl::Info> infos; |
||||
ocl::getDevice(infos); |
||||
|
||||
ocl::oclMat dsrc(src), dABFilter, dBFilter; |
||||
|
||||
Size ksize(ks, ks); |
||||
adaptiveBilateralFilter(src,abFilterCPU, ksize, 10); |
||||
ocl::adaptiveBilateralFilter(dsrc, dABFilter, ksize, 10); |
||||
ocl::bilateralFilter(dsrc, dBFilter, ks, 30, 9); |
||||
|
||||
Mat abFilter = dABFilter; |
||||
Mat bFilter = dBFilter; |
||||
imshow(winName[0], src); |
||||
|
||||
imshow(winName[1], abFilterCPU); |
||||
|
||||
imshow(winName[2], abFilter); |
||||
|
||||
imshow(winName[3], bFilter); |
||||
|
||||
waitKey(); |
||||
return 0; |
||||
|
||||
} |
@ -0,0 +1,136 @@ |
||||
#include <iostream> |
||||
#include <string> |
||||
|
||||
#include "opencv2/core.hpp" |
||||
#include "opencv2/core/utility.hpp" |
||||
#include "opencv2/ocl.hpp" |
||||
#include "opencv2/highgui.hpp" |
||||
|
||||
using namespace std; |
||||
using namespace cv; |
||||
using namespace cv::ocl; |
||||
|
||||
#define M_MOG 1 |
||||
#define M_MOG2 2 |
||||
|
||||
int main(int argc, const char** argv) |
||||
{ |
||||
|
||||
cv::CommandLineParser cmd(argc, argv, |
||||
"{ c camera | false | use camera }" |
||||
"{ f file | 768x576.avi | input video file }" |
||||
"{ m method | mog | method (mog, mog2) }" |
||||
"{ h help | false | print help message }"); |
||||
|
||||
if (cmd.get<bool>("help")) |
||||
{ |
||||
cout << "Usage : bgfg_segm [options]" << endl; |
||||
cout << "Avaible options:" << endl; |
||||
cmd.printMessage(); |
||||
return 0; |
||||
} |
||||
|
||||
bool useCamera = cmd.get<bool>("camera"); |
||||
string file = cmd.get<string>("file"); |
||||
string method = cmd.get<string>("method"); |
||||
|
||||
if (method != "mog" && method != "mog2") |
||||
{ |
||||
cerr << "Incorrect method" << endl; |
||||
return -1; |
||||
} |
||||
|
||||
int m = method == "mog" ? M_MOG : M_MOG2; |
||||
|
||||
VideoCapture cap; |
||||
|
||||
if (useCamera) |
||||
cap.open(0); |
||||
else |
||||
cap.open(file); |
||||
|
||||
if (!cap.isOpened()) |
||||
{ |
||||
cerr << "can not open camera or video file" << endl; |
||||
return -1; |
||||
} |
||||
|
||||
std::vector<cv::ocl::Info>info; |
||||
cv::ocl::getDevice(info); |
||||
|
||||
Mat frame; |
||||
cap >> frame; |
||||
|
||||
oclMat d_frame(frame); |
||||
|
||||
cv::ocl::MOG mog; |
||||
cv::ocl::MOG2 mog2; |
||||
|
||||
oclMat d_fgmask; |
||||
oclMat d_fgimg; |
||||
oclMat d_bgimg; |
||||
|
||||
d_fgimg.create(d_frame.size(), d_frame.type()); |
||||
|
||||
Mat fgmask; |
||||
Mat fgimg; |
||||
Mat bgimg; |
||||
|
||||
switch (m) |
||||
{ |
||||
case M_MOG: |
||||
mog(d_frame, d_fgmask, 0.01f); |
||||
break; |
||||
|
||||
case M_MOG2: |
||||
mog2(d_frame, d_fgmask); |
||||
break; |
||||
} |
||||
|
||||
for(;;) |
||||
{ |
||||
cap >> frame; |
||||
if (frame.empty()) |
||||
break; |
||||
d_frame.upload(frame); |
||||
|
||||
int64 start = cv::getTickCount(); |
||||
|
||||
//update the model
|
||||
switch (m) |
||||
{ |
||||
case M_MOG: |
||||
mog(d_frame, d_fgmask, 0.01f); |
||||
mog.getBackgroundImage(d_bgimg); |
||||
break; |
||||
|
||||
case M_MOG2: |
||||
mog2(d_frame, d_fgmask); |
||||
mog2.getBackgroundImage(d_bgimg); |
||||
break; |
||||
} |
||||
|
||||
double fps = cv::getTickFrequency() / (cv::getTickCount() - start); |
||||
std::cout << "FPS : " << fps << std::endl; |
||||
|
||||
d_fgimg.setTo(Scalar::all(0)); |
||||
d_frame.copyTo(d_fgimg, d_fgmask); |
||||
|
||||
d_fgmask.download(fgmask); |
||||
d_fgimg.download(fgimg); |
||||
if (!d_bgimg.empty()) |
||||
d_bgimg.download(bgimg); |
||||
|
||||
imshow("image", frame); |
||||
imshow("foreground mask", fgmask); |
||||
imshow("foreground image", fgimg); |
||||
if (!bgimg.empty()) |
||||
imshow("mean background image", bgimg); |
||||
|
||||
int key = waitKey(30); |
||||
if (key == 27) |
||||
break; |
||||
} |
||||
|
||||
return 0; |
||||
} |
Loading…
Reference in new issue