2015-03-24 19:11:44 +01:00
/***********************************************************************
* Software License Agreement ( BSD License )
*
* Copyright ( c ) 2015 Ippei Ito . All rights reserved .
*
* THE BSD LICENSE
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
*
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` 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 AUTHOR 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 .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2015-03-24 16:08:47 +01:00
/*
2015-03-24 19:11:44 +01:00
For OpenCV2 .4 / OpenCV3 .0
2015-03-24 16:08:47 +01:00
Test for Pull Request # 3829
https : //github.com/Itseez/opencv/pull/3829
This test code creates brute force matcher for accuracy of reference , and the test target matcher .
Then , add ( ) and train ( ) transformed query image descriptors , and some outlier images descriptors to both matchers .
Then , compared with the query image by match ( ) and findHomography ( ) to detect outlier and calculate accuracy .
And each drawMatches ( ) images are saved , if SAVE_DRAW_MATCHES_IMAGES is true .
Finally , compare accuracies between the brute force matcher and the test target matcher .
The lsh algorithm uses std : : random_shuffle in lsh_index . h to make the random indexes table .
So , in relation to default random seed value of the execution environment or by using " srand(time(0)) function " ,
the match time and accuracy of the match results are different , each time the code ran .
And the match time becomes late in relation to the number of the hash collision times .
*/
# include "test_precomp.hpp"
# include "opencv2/ts.hpp" // for FilePath::CreateFolder()
# include <time.h> // for time()
// If defined, the match time and accuracy of the match results are a little different, each time the code ran.
2015-03-24 21:16:30 +01:00
//#define INIT_RANDOM_SEED
2015-03-24 16:08:47 +01:00
// If defined, some outlier images descriptors add() the matcher.
# define TRAIN_WITH_OUTLIER_IMAGES
// If true, save drawMatches() images.
# define SAVE_DRAW_MATCHES_IMAGES false
// if true, verbose output
# define SHOW_DEBUG_LOG true
# if CV_MAJOR_VERSION==2
2015-04-02 12:58:19 +02:00
# define OrbCreate new cv::ORB(4000)
2015-03-24 16:08:47 +01:00
# elif CV_MAJOR_VERSION==3
2015-04-02 12:58:19 +02:00
# define OrbCreate cv::ORB::create(4000)
# define AKazeCreate cv::AKAZE::create()
2015-03-24 16:08:47 +01:00
# endif
using namespace std ;
int testno_for_make_filename = 0 ;
// --------------------------------------------------------------------------------------
// Parameter class to transform query image
// --------------------------------------------------------------------------------------
class testparam
{
public :
string transname ;
2015-04-02 12:58:19 +02:00
void ( * transfunc ) ( float , const cv : : Mat & , cv : : Mat & ) ;
2015-03-24 16:08:47 +01:00
float from , to , step ;
2015-04-02 12:58:19 +02:00
testparam ( string _transname , void ( * _transfunc ) ( float , const cv : : Mat & , cv : : Mat & ) , float _from , float _to , float _step ) :
2015-03-24 16:08:47 +01:00
transname ( _transname ) ,
transfunc ( _transfunc ) ,
from ( _from ) ,
to ( _to ) ,
step ( _step )
{ }
} ;
// --------------------------------------------------------------------------------------
// from matching_to_many_images.cpp
// --------------------------------------------------------------------------------------
2015-04-02 12:58:19 +02:00
int maskMatchesByTrainImgIdx ( const vector < cv : : DMatch > & matches , int trainImgIdx , vector < char > & mask )
2015-03-24 16:08:47 +01:00
{
int matchcnt = 0 ;
mask . resize ( matches . size ( ) ) ;
fill ( mask . begin ( ) , mask . end ( ) , 0 ) ;
for ( size_t i = 0 ; i < matches . size ( ) ; i + + )
{
if ( matches [ i ] . imgIdx = = trainImgIdx )
{
mask [ i ] = 1 ;
matchcnt + + ;
}
}
return matchcnt ;
}
2015-04-02 12:58:19 +02:00
int calcHomographyAndInlierCount ( const vector < cv : : KeyPoint > & query_kp , const vector < cv : : KeyPoint > & train_kp , const vector < cv : : DMatch > & match , vector < char > & mask , cv : : Mat & homography )
2015-03-24 16:08:47 +01:00
{
// make query and current train image keypoint pairs
std : : vector < cv : : Point2f > srcPoints , dstPoints ;
for ( unsigned int i = 0 ; i < match . size ( ) ; + + i )
{
if ( mask [ i ] ! = 0 ) // is current train image ?
{
srcPoints . push_back ( query_kp [ match [ i ] . queryIdx ] . pt ) ;
dstPoints . push_back ( train_kp [ match [ i ] . trainIdx ] . pt ) ;
}
}
// calc homography
vector < uchar > inlierMask ;
2015-04-02 12:58:19 +02:00
homography = findHomography ( srcPoints , dstPoints , cv : : RANSAC , 3.0 , inlierMask ) ;
2015-03-24 16:08:47 +01:00
// update outlier mask
int j = 0 ;
for ( unsigned int i = 0 ; i < match . size ( ) ; + + i )
{
if ( mask [ i ] ! = 0 ) // is current train image ?
{
if ( inlierMask . size ( ) = = 0 | | inlierMask [ j ] = = 0 ) // is outlier ?
{
mask [ i ] = 0 ;
}
j + + ;
}
}
// count inlier
int inlierCnt = 0 ;
for ( unsigned int i = 0 ; i < mask . size ( ) ; + + i )
{
if ( mask [ i ] ! = 0 )
{
inlierCnt + + ;
}
}
return inlierCnt ;
}
2015-04-02 12:58:19 +02:00
void drawDetectedRectangle ( cv : : Mat & imgResult , const cv : : Mat & homography , const cv : : Mat & imgQuery )
2015-03-24 16:08:47 +01:00
{
2015-04-02 12:58:19 +02:00
std : : vector < cv : : Point2f > query_corners ( 4 ) ;
query_corners [ 0 ] = cv : : Point ( 0 , 0 ) ;
query_corners [ 1 ] = cv : : Point ( imgQuery . cols , 0 ) ;
query_corners [ 2 ] = cv : : Point ( imgQuery . cols , imgQuery . rows ) ;
query_corners [ 3 ] = cv : : Point ( 0 , imgQuery . rows ) ;
std : : vector < cv : : Point2f > train_corners ( 4 ) ;
2015-03-24 16:08:47 +01:00
perspectiveTransform ( query_corners , train_corners , homography ) ;
2015-04-02 12:58:19 +02:00
line ( imgResult , train_corners [ 0 ] + query_corners [ 1 ] , train_corners [ 1 ] + query_corners [ 1 ] , cv : : Scalar ( 0 , 255 , 0 ) , 4 ) ;
line ( imgResult , train_corners [ 1 ] + query_corners [ 1 ] , train_corners [ 2 ] + query_corners [ 1 ] , cv : : Scalar ( 0 , 255 , 0 ) , 4 ) ;
line ( imgResult , train_corners [ 2 ] + query_corners [ 1 ] , train_corners [ 3 ] + query_corners [ 1 ] , cv : : Scalar ( 0 , 255 , 0 ) , 4 ) ;
line ( imgResult , train_corners [ 3 ] + query_corners [ 1 ] , train_corners [ 0 ] + query_corners [ 1 ] , cv : : Scalar ( 0 , 255 , 0 ) , 4 ) ;
2015-03-24 16:08:47 +01:00
}
// --------------------------------------------------------------------------------------
// transform query image, extract&compute, train, matching and save result image function
// --------------------------------------------------------------------------------------
typedef struct tagTrainInfo
{
int traindesccnt ;
double traintime ;
double matchtime ;
double accuracy ;
} TrainInfo ;
TrainInfo transImgAndTrain (
2015-04-02 12:58:19 +02:00
cv : : Feature2D * fe ,
cv : : DescriptorMatcher * matcher ,
2015-03-24 16:08:47 +01:00
const string & matchername ,
2015-04-02 12:58:19 +02:00
const cv : : Mat & imgQuery , const vector < cv : : KeyPoint > & query_kp , const cv : : Mat & query_desc ,
const vector < cv : : Mat > & imgOutliers , const vector < vector < cv : : KeyPoint > > & outliers_kp , const vector < cv : : Mat > & outliers_desc , const int totalOutlierDescCnt ,
2015-03-24 21:16:30 +01:00
const float t , const testparam * tp ,
2015-03-24 16:08:47 +01:00
const int testno , const bool bVerboseOutput , const bool bSaveDrawMatches )
{
TrainInfo ti ;
// transform query image
2015-04-02 12:58:19 +02:00
cv : : Mat imgTransform ;
2015-03-24 21:16:30 +01:00
( tp - > transfunc ) ( t , imgQuery , imgTransform ) ;
2015-03-24 16:08:47 +01:00
// extract kp and compute desc from transformed query image
2015-04-02 12:58:19 +02:00
vector < cv : : KeyPoint > trans_query_kp ;
cv : : Mat trans_query_desc ;
2015-03-24 16:08:47 +01:00
# if CV_MAJOR_VERSION==2
2015-04-02 12:58:19 +02:00
( * fe ) ( imgTransform , cv : : Mat ( ) , trans_query_kp , trans_query_desc ) ;
2015-03-24 16:08:47 +01:00
# elif CV_MAJOR_VERSION==3
fe - > detectAndCompute ( imgTransform , Mat ( ) , trans_query_kp , trans_query_desc ) ;
# endif
// add&train transformed query desc and outlier desc
matcher - > clear ( ) ;
2015-04-02 12:58:19 +02:00
matcher - > add ( vector < cv : : Mat > ( 1 , trans_query_desc ) ) ;
double s = ( double ) cv : : getTickCount ( ) ;
2015-03-24 16:08:47 +01:00
matcher - > train ( ) ;
2015-04-02 12:58:19 +02:00
ti . traintime = 1000.0 * ( ( double ) cv : : getTickCount ( ) - s ) / cv : : getTickFrequency ( ) ;
2015-03-24 16:08:47 +01:00
ti . traindesccnt = trans_query_desc . rows ;
# if defined(TRAIN_WITH_OUTLIER_IMAGES)
// same as matcher->add(outliers_desc); matcher->train();
for ( unsigned int i = 0 ; i < outliers_desc . size ( ) ; + + i )
{
2015-04-02 12:58:19 +02:00
matcher - > add ( vector < cv : : Mat > ( 1 , outliers_desc [ i ] ) ) ;
s = ( double ) cv : : getTickCount ( ) ;
2015-03-24 16:08:47 +01:00
matcher - > train ( ) ;
2015-04-02 12:58:19 +02:00
ti . traintime + = 1000.0 * ( ( double ) cv : : getTickCount ( ) - s ) / cv : : getTickFrequency ( ) ;
2015-03-24 16:08:47 +01:00
}
ti . traindesccnt + = totalOutlierDescCnt ;
# endif
// matching
2015-04-02 12:58:19 +02:00
vector < cv : : DMatch > match ;
s = ( double ) cv : : getTickCount ( ) ;
2015-03-24 16:08:47 +01:00
matcher - > match ( query_desc , match ) ;
2015-04-02 12:58:19 +02:00
ti . matchtime = 1000.0 * ( ( double ) cv : : getTickCount ( ) - s ) / cv : : getTickFrequency ( ) ;
2015-03-24 16:08:47 +01:00
// prepare a directory and variables for save matching images
vector < char > mask ;
2015-04-02 12:58:19 +02:00
cv : : Mat imgResult ;
2015-03-24 16:08:47 +01:00
const char resultDir [ ] = " result " ;
if ( bSaveDrawMatches )
{
testing : : internal : : FilePath fp = testing : : internal : : FilePath ( resultDir ) ;
fp . CreateFolder ( ) ;
}
char buff [ 2048 ] ;
int matchcnt ;
// save query vs transformed query matching image with detected rectangle
matchcnt = maskMatchesByTrainImgIdx ( match , ( int ) 0 , mask ) ;
// calc homography and inlier
2015-04-02 12:58:19 +02:00
cv : : Mat homography ;
2015-03-24 16:08:47 +01:00
int inlierCnt = calcHomographyAndInlierCount ( query_kp , trans_query_kp , match , mask , homography ) ;
ti . accuracy = ( double ) inlierCnt / ( double ) mask . size ( ) * 100.0 ;
2015-04-02 12:58:19 +02:00
drawMatches ( imgQuery , query_kp , imgTransform , trans_query_kp , match , imgResult , cv : : Scalar : : all ( - 1 ) , cv : : Scalar : : all ( 128 ) , mask , cv : : DrawMatchesFlags : : DRAW_RICH_KEYPOINTS ) ;
2015-03-24 16:08:47 +01:00
if ( inlierCnt )
{
// draw detected rectangle
drawDetectedRectangle ( imgResult , homography , imgQuery ) ;
}
// draw status
sprintf ( buff , " %s accuracy:%-3.2f%% %d descriptors training time:%-3.2fms matching :%-3.2fms " , matchername . c_str ( ) , ti . accuracy , ti . traindesccnt , ti . traintime , ti . matchtime ) ;
2015-04-02 12:58:19 +02:00
putText ( imgResult , buff , cv : : Point ( 0 , 12 ) , cv : : FONT_HERSHEY_PLAIN , 0.8 , cv : : Scalar ( 0. , 0. , 255. ) ) ;
2015-03-24 21:16:30 +01:00
sprintf ( buff , " %s/res%03d_%s_%s%.1f_inlier.png " , resultDir , testno , matchername . c_str ( ) , tp - > transname . c_str ( ) , t ) ;
2015-03-24 16:08:47 +01:00
if ( bSaveDrawMatches & & ! imwrite ( buff , imgResult ) ) cout < < " Image " < < buff < < " can not be saved (may be because directory " < < resultDir < < " does not exist). " < < endl ;
# if defined(TRAIN_WITH_OUTLIER_IMAGES)
// save query vs outlier matching image(s)
for ( unsigned int i = 0 ; i < imgOutliers . size ( ) ; + + i )
{
matchcnt = maskMatchesByTrainImgIdx ( match , ( int ) i + 1 , mask ) ;
2015-04-02 12:58:19 +02:00
drawMatches ( imgQuery , query_kp , imgOutliers [ i ] , outliers_kp [ i ] , match , imgResult , cv : : Scalar : : all ( - 1 ) , cv : : Scalar : : all ( 128 ) , mask ) ; // , DrawMatchesFlags::DRAW_RICH_KEYPOINTS);
2015-03-24 16:08:47 +01:00
sprintf ( buff , " query_num:%d train_num:%d matched:%d %d descriptors training time:%-3.2fms matching :%-3.2fms " , ( int ) query_kp . size ( ) , ( int ) outliers_kp [ i ] . size ( ) , matchcnt , ti . traindesccnt , ti . traintime , ti . matchtime ) ;
2015-04-02 12:58:19 +02:00
putText ( imgResult , buff , cv : : Point ( 0 , 12 ) , cv : : FONT_HERSHEY_PLAIN , 0.8 , cv : : Scalar ( 0. , 0. , 255. ) ) ;
2015-03-24 21:16:30 +01:00
sprintf ( buff , " %s/res%03d_%s_%s%.1f_outlier%02d.png " , resultDir , testno , matchername . c_str ( ) , tp - > transname . c_str ( ) , t , i ) ;
2015-03-24 16:08:47 +01:00
if ( bSaveDrawMatches & & ! imwrite ( buff , imgResult ) ) cout < < " Image " < < buff < < " can not be saved (may be because directory " < < resultDir < < " does not exist). " < < endl ;
}
# endif
if ( bVerboseOutput )
{
2015-03-24 21:16:30 +01:00
cout < < tp - > transname < < " image matching accuracy: " < < ti . accuracy < < " % " < < ti . traindesccnt < < " train: " < < ti . traintime < < " ms match: " < < ti . matchtime < < " ms " < < endl ;
2015-03-24 16:08:47 +01:00
}
return ti ;
}
// --------------------------------------------------------------------------------------
// Main Test Class
// --------------------------------------------------------------------------------------
class CV_FeatureDetectorMatcherBaseTest : public cvtest : : BaseTest
{
private :
2015-03-25 10:24:52 +01:00
testparam * tp ;
double target_accuracy_margin_from_bfmatcher ;
2015-04-02 12:58:19 +02:00
cv : : Feature2D * fe ; // feature detector extractor
cv : : DescriptorMatcher * bfmatcher ; // brute force matcher for accuracy of reference
cv : : DescriptorMatcher * flmatcher ; // flann matcher to test
cv : : Mat imgQuery ; // query image
vector < cv : : Mat > imgOutliers ; // outlier image
vector < cv : : KeyPoint > query_kp ; // query key points detect from imgQuery
cv : : Mat query_desc ; // query descriptors extract from imgQuery
vector < vector < cv : : KeyPoint > > outliers_kp ;
vector < cv : : Mat > outliers_desc ;
2015-03-24 16:08:47 +01:00
int totalOutlierDescCnt ;
string flmatchername ;
public :
//
// constructor
//
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest ( testparam * _tp , double _accuracy_margin , cv : : Feature2D * _fe , cv : : DescriptorMatcher * _flmatcher , string _flmatchername , int norm_type_for_bfmatcher ) :
2015-03-24 16:08:47 +01:00
tp ( _tp ) ,
2015-03-25 10:24:52 +01:00
target_accuracy_margin_from_bfmatcher ( _accuracy_margin ) ,
2015-03-24 16:08:47 +01:00
fe ( _fe ) ,
flmatcher ( _flmatcher ) ,
2015-03-25 10:24:52 +01:00
flmatchername ( _flmatchername )
2015-03-24 16:08:47 +01:00
{
# if defined(INIT_RANDOM_SEED)
// from test/test_eigen.cpp
srand ( ( unsigned int ) time ( 0 ) ) ;
# endif
// create brute force matcher for accuracy of reference
2015-04-02 12:58:19 +02:00
bfmatcher = new cv : : BFMatcher ( norm_type_for_bfmatcher ) ;
2015-03-24 16:08:47 +01:00
}
//
// Main Test method
//
virtual void run ( int )
{
// load query image
string strQueryFile = string ( cvtest : : TS : : ptr ( ) - > get_data_path ( ) ) + " shared/lena.png " ;
2015-04-02 12:58:19 +02:00
imgQuery = cv : : imread ( strQueryFile , 0 ) ;
2015-03-24 16:08:47 +01:00
if ( imgQuery . empty ( ) )
{
ts - > printf ( cvtest : : TS : : LOG , " Image %s can not be read. \n " , strQueryFile . c_str ( ) ) ;
ts - > set_failed_test_info ( cvtest : : TS : : FAIL_INVALID_TEST_DATA ) ;
return ;
}
// load outlier images
char * outliers [ ] = { ( char * ) " baboon.png " , ( char * ) " fruits.png " , ( char * ) " airplane.png " } ;
for ( unsigned int i = 0 ; i < sizeof ( outliers ) / sizeof ( char * ) ; i + + )
{
string strOutlierFile = string ( cvtest : : TS : : ptr ( ) - > get_data_path ( ) ) + " shared/ " + outliers [ i ] ;
2015-04-02 12:58:19 +02:00
cv : : Mat imgOutlier = cv : : imread ( strOutlierFile , 0 ) ;
2015-03-24 16:08:47 +01:00
if ( imgQuery . empty ( ) )
{
ts - > printf ( cvtest : : TS : : LOG , " Image %s can not be read. \n " , strOutlierFile . c_str ( ) ) ;
ts - > set_failed_test_info ( cvtest : : TS : : FAIL_INVALID_TEST_DATA ) ;
return ;
}
imgOutliers . push_back ( imgOutlier ) ;
}
// extract and compute keypoints and descriptors from query image
# if CV_MAJOR_VERSION==2
2015-04-02 12:58:19 +02:00
( * fe ) ( imgQuery , cv : : Mat ( ) , query_kp , query_desc ) ;
2015-03-24 16:08:47 +01:00
# elif CV_MAJOR_VERSION==3
fe - > detectAndCompute ( imgQuery , Mat ( ) , query_kp , query_desc ) ;
# endif
// extract and compute keypoints and descriptors from outlier images
fe - > detect ( imgOutliers , outliers_kp ) ;
2015-04-02 12:58:19 +02:00
( ( cv : : DescriptorExtractor * ) fe ) - > compute ( imgOutliers , outliers_kp , outliers_desc ) ;
2015-03-24 16:08:47 +01:00
totalOutlierDescCnt = 0 ;
for ( unsigned int i = 0 ; i < outliers_desc . size ( ) ; + + i ) totalOutlierDescCnt + = outliers_desc [ i ] . rows ;
if ( SHOW_DEBUG_LOG )
{
cout < < query_kp . size ( ) < < " keypoints extracted from query image. " < < endl ;
# if defined(TRAIN_WITH_OUTLIER_IMAGES)
cout < < totalOutlierDescCnt < < " keypoints extracted from outlier image(s). " < < endl ;
# endif
}
// compute brute force matcher accuracy for reference
double totalTrainTime = 0. ;
double totalMatchTime = 0. ;
double totalAccuracy = 0. ;
int cnt = 0 ;
2015-03-24 21:16:30 +01:00
for ( float t = tp - > from ; t < = tp - > to ; t + = tp - > step , + + testno_for_make_filename , + + cnt )
2015-03-24 16:08:47 +01:00
{
if ( SHOW_DEBUG_LOG ) cout < < " Test No. " < < testno_for_make_filename < < " BFMatcher " < < t ;
TrainInfo ti = transImgAndTrain ( fe , bfmatcher , " BFMatcher " ,
imgQuery , query_kp , query_desc ,
imgOutliers , outliers_kp , outliers_desc ,
totalOutlierDescCnt ,
t , tp , testno_for_make_filename , SHOW_DEBUG_LOG , SAVE_DRAW_MATCHES_IMAGES ) ;
totalTrainTime + = ti . traintime ;
totalMatchTime + = ti . matchtime ;
totalAccuracy + = ti . accuracy ;
}
double bf_average_accuracy = totalAccuracy / cnt ;
if ( SHOW_DEBUG_LOG )
{
cout < < " total training time: " < < totalTrainTime < < " ms " < < endl ;
cout < < " total matching time: " < < totalMatchTime < < " ms " < < endl ;
cout < < " average accuracy: " < < bf_average_accuracy < < " % " < < endl ;
}
// test the target matcher
totalTrainTime = 0. ;
totalMatchTime = 0. ;
totalAccuracy = 0. ;
cnt = 0 ;
2015-03-24 21:16:30 +01:00
for ( float t = tp - > from ; t < = tp - > to ; t + = tp - > step , + + testno_for_make_filename , + + cnt )
2015-03-24 16:08:47 +01:00
{
if ( SHOW_DEBUG_LOG ) cout < < " Test No. " < < testno_for_make_filename < < " " < < flmatchername < < " " < < t ;
TrainInfo ti = transImgAndTrain ( fe , flmatcher , flmatchername ,
imgQuery , query_kp , query_desc ,
imgOutliers , outliers_kp , outliers_desc ,
totalOutlierDescCnt ,
t , tp , testno_for_make_filename , SHOW_DEBUG_LOG , SAVE_DRAW_MATCHES_IMAGES ) ;
totalTrainTime + = ti . traintime ;
totalMatchTime + = ti . matchtime ;
totalAccuracy + = ti . accuracy ;
}
double average_accuracy = totalAccuracy / cnt ;
double target_average_accuracy = bf_average_accuracy * target_accuracy_margin_from_bfmatcher ;
if ( SHOW_DEBUG_LOG )
{
cout < < " total training time: " < < totalTrainTime < < " ms " < < endl ;
cout < < " total matching time: " < < totalMatchTime < < " ms " < < endl ;
cout < < " average accuracy: " < < average_accuracy < < " % " < < endl ;
cout < < " threshold of the target matcher average accuracy as error : " < < target_average_accuracy < < " % " < < endl ;
cout < < " accuracy degraded " < < ( 100.0 - ( average_accuracy / bf_average_accuracy * 100.0 ) ) < < " % from BFMatcher.(lower percentage is better) " < < endl ;
}
// compare accuracies between the brute force matcher and the test target matcher
if ( average_accuracy < target_average_accuracy )
{
2015-03-24 21:16:30 +01:00
ts - > printf ( cvtest : : TS : : LOG , " Bad average accuracy %f < %f while test %s %s query \n " , average_accuracy , target_average_accuracy , flmatchername . c_str ( ) , tp - > transname . c_str ( ) ) ;
2015-03-24 16:08:47 +01:00
ts - > set_failed_test_info ( cvtest : : TS : : FAIL_BAD_ACCURACY ) ;
}
return ;
}
} ;
// --------------------------------------------------------------------------------------
// Transform Functions
// --------------------------------------------------------------------------------------
2015-04-02 12:58:19 +02:00
static void rotate ( float deg , const cv : : Mat & src , cv : : Mat & dst )
2015-03-24 16:08:47 +01:00
{
2015-04-02 12:58:19 +02:00
cv : : warpAffine ( src , dst , getRotationMatrix2D ( cv : : Point2f ( src . cols / 2.0f , src . rows / 2.0f ) , deg , 1 ) , src . size ( ) , cv : : INTER_CUBIC ) ;
2015-03-24 16:08:47 +01:00
}
2015-04-02 12:58:19 +02:00
static void scale ( float scale , const cv : : Mat & src , cv : : Mat & dst )
2015-03-24 16:08:47 +01:00
{
2015-04-02 12:58:19 +02:00
cv : : resize ( src , dst , cv : : Size ( ( int ) ( src . cols * scale ) , ( int ) ( src . rows * scale ) ) , cv : : INTER_CUBIC ) ;
2015-03-24 16:08:47 +01:00
}
2015-04-02 12:58:19 +02:00
static void blur ( float k , const cv : : Mat & src , cv : : Mat & dst )
2015-03-24 16:08:47 +01:00
{
2015-04-02 12:58:19 +02:00
GaussianBlur ( src , dst , cv : : Size ( ( int ) k , ( int ) k ) , 0 ) ;
2015-03-24 16:08:47 +01:00
}
// --------------------------------------------------------------------------------------
// Tests Registrations
// --------------------------------------------------------------------------------------
# define SHORT_LSH_KEY_ACCURACY_MARGIN 0.72 // The margin for FlannBasedMatcher. 28% degraded from BFMatcher(Actually, about 10..24% measured.lower percentage is better.) for lsh key size=16.
# define MIDDLE_LSH_KEY_ACCURACY_MARGIN 0.72 // The margin for FlannBasedMatcher. 28% degraded from BFMatcher(Actually, about 7..24% measured.lower percentage is better.) for lsh key size=24.
# define LONG_LSH_KEY_ACCURACY_MARGIN 0.90 // The margin for FlannBasedMatcher. 10% degraded from BFMatcher(Actually, about -29...7% measured.lower percentage is better.) for lsh key size=31.
TEST ( BlurredQueryFlannBasedLshShortKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 16 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " blurred " , blur , 1.0f , 11.0f , 2.0f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , SHORT_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 16, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( BlurredQueryFlannBasedLshMiddleKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 24 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " blurred " , blur , 1.0f , 11.0f , 2.0f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , MIDDLE_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 24, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( BlurredQueryFlannBasedLshLongKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 31 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " blurred " , blur , 1.0f , 11.0f , 2.0f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , LONG_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 31, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( ScaledQueryFlannBasedLshShortKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 16 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " scaled " , scale , 0.5f , 1.5f , 0.1f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , SHORT_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 16, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( ScaledQueryFlannBasedLshMiddleKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 24 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " scaled " , scale , 0.5f , 1.5f , 0.1f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , MIDDLE_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 24, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( ScaledQueryFlannBasedLshLongKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 31 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " scaled " , scale , 0.5f , 1.5f , 0.1f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , LONG_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 31, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( RotatedQueryFlannBasedLshShortKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 16 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " rotated " , rotate , 0.0f , 359.0f , 30.0f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , SHORT_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 16, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( RotatedQueryFlannBasedLshMiddleKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 24 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " rotated " , rotate , 0.0f , 359.0f , 30.0f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , MIDDLE_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 24, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}
TEST ( RotatedQueryFlannBasedLshLongKeyMatcherAdditionalTrainTest , accuracy )
{
2015-04-02 12:58:19 +02:00
cv : : Ptr < cv : : Feature2D > fe = OrbCreate ;
cv : : Ptr < cv : : FlannBasedMatcher > fl = cv : : makePtr < cv : : FlannBasedMatcher > ( cv : : makePtr < cv : : flann : : LshIndexParams > ( 1 , 31 , 2 ) ) ;
2015-03-24 16:08:47 +01:00
testparam tp ( " rotated " , rotate , 0.0f , 359.0f , 30.0f ) ;
2015-04-02 12:58:19 +02:00
CV_FeatureDetectorMatcherBaseTest test ( & tp , LONG_LSH_KEY_ACCURACY_MARGIN , fe , fl , " FlannLsh(1, 31, 2) " , cv::NORM_HAMMING) ;
2015-03-24 16:08:47 +01:00
test . safe_run ( ) ;
}