/*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. // // // Intel License Agreement // For Open Source Computer Vision Library // // Copyright (C) 2000, Intel Corporation, all rights reserved. // Third party copyrights are property of their respective owners. // // 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 materials provided with the distribution. // // * The name of Intel Corporation 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 "_cxts.h" #include #include #include #include #if defined WIN32 || defined _WIN32 || defined WIN64 || defined _WIN64 #include #else #include #endif CvTest* CvTest::first = 0; CvTest* CvTest::last = 0; int CvTest::test_count = 0; /*****************************************************************************************\ * Exception and memory handlers * \*****************************************************************************************/ // a few platform-dependent declarations #define CV_TS_NORMAL 0 #define CV_TS_BLUE 1 #define CV_TS_GREEN 2 #define CV_TS_RED 4 #if defined WIN32 || defined _WIN32 || defined WIN64 || defined _WIN64 #include #ifdef _MSC_VER #include #endif #ifdef _MSC_VER static void cv_seh_translator( unsigned int /*u*/, EXCEPTION_POINTERS* pExp ) { int code = CvTS::FAIL_EXCEPTION; switch( pExp->ExceptionRecord->ExceptionCode ) { case EXCEPTION_ACCESS_VIOLATION: case EXCEPTION_ARRAY_BOUNDS_EXCEEDED: case EXCEPTION_DATATYPE_MISALIGNMENT: case EXCEPTION_FLT_STACK_CHECK: case EXCEPTION_STACK_OVERFLOW: case EXCEPTION_IN_PAGE_ERROR: code = CvTS::FAIL_MEMORY_EXCEPTION; break; case EXCEPTION_FLT_DENORMAL_OPERAND: case EXCEPTION_FLT_DIVIDE_BY_ZERO: case EXCEPTION_FLT_INEXACT_RESULT: case EXCEPTION_FLT_INVALID_OPERATION: case EXCEPTION_FLT_OVERFLOW: case EXCEPTION_FLT_UNDERFLOW: case EXCEPTION_INT_DIVIDE_BY_ZERO: case EXCEPTION_INT_OVERFLOW: code = CvTS::FAIL_ARITHM_EXCEPTION; break; case EXCEPTION_BREAKPOINT: case EXCEPTION_ILLEGAL_INSTRUCTION: case EXCEPTION_INVALID_DISPOSITION: case EXCEPTION_NONCONTINUABLE_EXCEPTION: case EXCEPTION_PRIV_INSTRUCTION: case EXCEPTION_SINGLE_STEP: code = CvTS::FAIL_EXCEPTION; } throw code; } #endif static void change_color( int color ) { static int normal_attributes = -1; HANDLE hstdout = GetStdHandle(STD_OUTPUT_HANDLE); fflush(stdout); if( normal_attributes < 0 ) { CONSOLE_SCREEN_BUFFER_INFO info; GetConsoleScreenBufferInfo( hstdout, &info ); normal_attributes = info.wAttributes; } SetConsoleTextAttribute( hstdout, (WORD)(color == CV_TS_NORMAL ? normal_attributes : ((color & CV_TS_BLUE ? FOREGROUND_BLUE : 0)| (color & CV_TS_GREEN ? FOREGROUND_GREEN : 0)| (color & CV_TS_RED ? FOREGROUND_RED : 0)|FOREGROUND_INTENSITY)) ); } #else #include static const int cv_ts_sig_id[] = { SIGSEGV, SIGBUS, SIGFPE, SIGILL, SIGABRT, -1 }; static jmp_buf cv_ts_jmp_mark; void cv_signal_handler( int sig_code ) { int code = CvTS::FAIL_EXCEPTION; switch( sig_code ) { case SIGFPE: code = CvTS::FAIL_ARITHM_EXCEPTION; break; case SIGSEGV: case SIGBUS: code = CvTS::FAIL_ARITHM_EXCEPTION; break; case SIGILL: code = CvTS::FAIL_EXCEPTION; } longjmp( cv_ts_jmp_mark, code ); } static void change_color( int color ) { static const uchar ansi_tab[] = { 30, 34, 32, 36, 31, 35, 33, 37 }; char buf[16]; int code = 0; fflush( stdout ); if( color != CV_TS_NORMAL ) code = ansi_tab[color & (CV_TS_BLUE|CV_TS_GREEN|CV_TS_RED)]; sprintf( buf, "\x1b[%dm", code ); fputs( buf, stdout ); } #endif // reads 16-digit hexadecimal number (i.e. 64-bit integer) static int64 read_seed( const char* str ) { int64 val = 0; if( str && strlen(str) == 16 ) { for( int i = 0; str[i]; i++ ) { int c = tolower(str[i]); if( !isxdigit(c) ) return 0; val = val * 16 + (str[i] < 'a' ? str[i] - '0' : str[i] - 'a' + 10); } } return val; } /***************************** memory manager *****************************/ typedef struct CvTestAllocBlock { struct CvTestAllocBlock* prev; struct CvTestAllocBlock* next; char* origin; char* data; size_t size; int index; } CvTestAllocBlock; class CvTestMemoryManager { public: CvTestMemoryManager( CvTS* ts ); virtual ~CvTestMemoryManager(); virtual void clear_and_check( int min_index = -1 ); virtual void start_tracking( int index_to_stop_at=-1 ); virtual void stop_tracking_and_check(); int get_alloc_index() { return index; } static void* alloc_proxy( size_t size, void* userdata ); static int free_proxy( void* ptr, void* userdata ); protected: virtual void* alloc( size_t size ); virtual int free( void* ptr ); virtual int free_block( CvTestAllocBlock* block ); int index; int track_blocks; int show_msg_box; int index_to_stop_at; const char* guard_pattern; int guard_size; int block_align; enum { MAX_MARKS = 1024 }; int marks[MAX_MARKS]; int marks_top; CvTS* ts; CvTestAllocBlock* first; CvTestAllocBlock* last; }; void* CvTestMemoryManager::alloc_proxy( size_t size, void* userdata ) { return ((CvTestMemoryManager*)userdata)->alloc( size ); } int CvTestMemoryManager::free_proxy( void* ptr, void* userdata ) { return ((CvTestMemoryManager*)userdata)->free( ptr ); } CvTestMemoryManager::CvTestMemoryManager( CvTS* _test_system ) { ts = _test_system; guard_pattern = "THIS IS A GUARD PATTERN!"; guard_size = (int)strlen(guard_pattern); block_align = CV_MALLOC_ALIGN; track_blocks = 0; marks_top = 0; first = last = 0; index = 0; index_to_stop_at = -1; show_msg_box = 1; } CvTestMemoryManager::~CvTestMemoryManager() { clear_and_check(); } void CvTestMemoryManager::clear_and_check( int min_index ) { int alloc_index = -1; CvTestAllocBlock* block; int leak_size = 0, leak_block_count = 0, mem_size = 0; void* mem_addr = 0; while( marks_top > 0 && marks[marks_top - 1] >= min_index ) marks_top--; for( block = last; block != 0; ) { CvTestAllocBlock* prev = block->prev; if( block->index < min_index ) break; leak_size += (int)block->size; leak_block_count++; alloc_index = block->index; mem_addr = block->data; mem_size = (int)block->size; free_block( block ); block = prev; } track_blocks--; if( leak_block_count > 0 ) { ts->set_failed_test_info( CvTS::FAIL_MEMORY_LEAK, alloc_index ); ts->printf( CvTS::LOG, "Memory leaks: %u blocks, %u bytes total\n" "%s leaked block: %p, %u bytes\n", leak_block_count, leak_size, leak_block_count > 1 ? "The first" : "The", mem_addr, mem_size ); } index = block ? block->index + 1 : 0; } void CvTestMemoryManager::start_tracking( int _index_to_stop_at ) { track_blocks--; marks[marks_top++] = index; assert( marks_top <= MAX_MARKS ); track_blocks+=2; index_to_stop_at = _index_to_stop_at >= index ? _index_to_stop_at : -1; } void CvTestMemoryManager::stop_tracking_and_check() { if( marks_top > 0 ) { int min_index = marks[--marks_top]; clear_and_check( min_index ); } } int CvTestMemoryManager::free_block( CvTestAllocBlock* block ) { int code = 0; char* data = block->data; if( block->origin == 0 || ((size_t)block->origin & (sizeof(double)-1)) != 0 ) code = CvTS::FAIL_MEMORY_CORRUPTION_BEGIN; if( memcmp( data - guard_size, guard_pattern, guard_size ) != 0 ) code = CvTS::FAIL_MEMORY_CORRUPTION_BEGIN; else if( memcmp( data + block->size, guard_pattern, guard_size ) != 0 ) code = CvTS::FAIL_MEMORY_CORRUPTION_END; if( code >= 0 ) { if( block->prev ) block->prev->next = block->next; else if( first == block ) first = block->next; if( block->next ) block->next->prev = block->prev; else if( last == block ) last = block->prev; free( block->origin ); } else { ts->set_failed_test_info( code, block->index ); ts->printf( CvTS::LOG, "Corrupted block (%s): %p, %u bytes\n", code == CvTS::FAIL_MEMORY_CORRUPTION_BEGIN ? "beginning" : "end", block->data, block->size ); } return code; } void* CvTestMemoryManager::alloc( size_t size ) { char* data; CvTestAllocBlock* block; size_t new_size = sizeof(*block) + size + guard_size*2 + block_align + sizeof(size_t)*2; char* ptr = (char*)malloc( new_size ); if( !ptr ) return 0; data = (char*)cvAlignPtr( ptr + sizeof(size_t) + sizeof(*block) + guard_size, block_align ); block = (CvTestAllocBlock*)cvAlignPtr( data - guard_size - sizeof(size_t) - sizeof(*block), sizeof(size_t) ); block->origin = ptr; block->data = data; block->size = 0; block->index = -1; block->next = block->prev = 0; memcpy( data - guard_size, guard_pattern, guard_size ); memcpy( data + size, guard_pattern, guard_size ); if( track_blocks > 0 ) { track_blocks--; block->size = size; if( index == index_to_stop_at ) { if( show_msg_box ) { #if defined WIN32 || defined _WIN32 MessageBox( NULL, "The block that is corrupted and/or not deallocated has been just allocated\n" "Press Ok to start debugging", "Memory Manager", MB_ICONERROR|MB_OK|MB_SYSTEMMODAL ); #endif } CV_DBG_BREAK(); } block->index = index++; block->prev = last; block->next = 0; if( last ) last = last->next = block; else first = last = block; track_blocks++; } return data; } int CvTestMemoryManager::free( void* ptr ) { char* data = (char*)ptr; CvTestAllocBlock* block = (CvTestAllocBlock*) cvAlignPtr( data - guard_size - sizeof(size_t) - sizeof(*block), sizeof(size_t) ); int code = free_block( block ); if( code < 0 && ts->is_debug_mode() ) CV_DBG_BREAK(); return 0; } /***************************** error handler *****************************/ #if 0 static int cvTestErrorCallback( int status, const char* func_name, const char* err_msg, const char* file_name, int line, void* userdata ) { if( status < 0 && status != CV_StsBackTrace && status != CV_StsAutoTrace ) ((CvTS*)userdata)->set_failed_test_info( CvTS::FAIL_ERROR_IN_CALLED_FUNC ); // print error message return cvStdErrReport( status, func_name, err_msg, file_name, line, 0 ); } #endif /*****************************************************************************************\ * Base Class for Tests * \*****************************************************************************************/ CvTest::CvTest( const char* _test_name, const char* _test_funcs, const char* _test_descr ) : name(_test_name ? _test_name : ""), tested_functions(_test_funcs ? _test_funcs : ""), description(_test_descr ? _test_descr : ""), ts(0) { if( last ) last->next = this; else first = this; last = this; test_count++; ts = 0; hdr_state = 0; timing_param_names = 0; timing_param_current = 0; timing_param_seqs = 0; timing_param_idxs = 0; timing_param_count = -1; test_case_count = -1; support_testing_modes = CvTS::CORRECTNESS_CHECK_MODE; } CvTest::~CvTest() { clear(); } CvTest* CvTest::get_first_test() { return first; } void CvTest::clear() { if( timing_param_current ) free( timing_param_current ); if( timing_param_seqs ) free( timing_param_seqs ); if( timing_param_idxs ) free( timing_param_idxs ); timing_param_current = 0; timing_param_seqs = 0; timing_param_idxs = 0; timing_param_count = -1; } int CvTest::init( CvTS* _test_system ) { clear(); ts = _test_system; return read_params( ts->get_file_storage() ); } const char* CvTest::get_parent_name( const char* name, char* buffer ) { const char* dash_pos = strrchr( name ? name : "", '-' ); if( !dash_pos ) return 0; if( name != (const char*)buffer ) strncpy( buffer, name, dash_pos - name ); buffer[dash_pos - name] = '\0'; return buffer; } const CvFileNode* CvTest::find_param( CvFileStorage* fs, const char* param_name ) { char buffer[256]; const char* name = get_name(); CvFileNode* node = 0; for(;;) { if( !name ) break; node = cvGetFileNodeByName( fs, 0, name ); if( node ) { node = cvGetFileNodeByName( fs, node, param_name ); if( node ) break; } name = get_parent_name( name, buffer ); } return node; } void CvTest::start_write_param( CvFileStorage* fs ) { if( hdr_state == 0 ) { cvStartWriteStruct( fs, get_name(), CV_NODE_MAP ); hdr_state = 1; } } void CvTest::write_param( CvFileStorage* fs, const char* paramname, int val ) { if( !ts->find_written_param( this, paramname, CV_NODE_INT, &val) ) { start_write_param( fs ); cvWriteInt( fs, paramname, val ); } } void CvTest::write_param( CvFileStorage* fs, const char* paramname, double val ) { if( !ts->find_written_param( this, paramname, CV_NODE_REAL, &val) ) { start_write_param( fs ); cvWriteReal( fs, paramname, val ); } } void CvTest::write_param( CvFileStorage* fs, const char* paramname, const char* val ) { if( !ts->find_written_param( this, paramname, CV_NODE_STRING, &val) ) { start_write_param( fs ); cvWriteString( fs, paramname, val ); } } void CvTest::write_string_list( CvFileStorage* fs, const char* paramname, const char** val, int count ) { if( val ) { start_write_param( fs ); int i; if( count < 0 ) count = INT_MAX; cvStartWriteStruct( fs, paramname, CV_NODE_SEQ + CV_NODE_FLOW ); for( i = 0; i < count && val[i] != 0; i++ ) cvWriteString( fs, 0, val[i] ); cvEndWriteStruct( fs ); } } void CvTest::write_int_list( CvFileStorage* fs, const char* paramname, const int* val, int count, int stop_value ) { if( val ) { start_write_param( fs ); int i; if( count < 0 ) count = INT_MAX; cvStartWriteStruct( fs, paramname, CV_NODE_SEQ + CV_NODE_FLOW ); for( i = 0; i < count && val[i] != stop_value; i++ ) cvWriteInt( fs, 0, val[i] ); cvEndWriteStruct( fs ); } } void CvTest::write_real_list( CvFileStorage* fs, const char* paramname, const double* val, int count, double stop_value ) { if( val ) { start_write_param( fs ); int i; if( count < 0 ) count = INT_MAX; cvStartWriteStruct( fs, paramname, CV_NODE_SEQ + CV_NODE_FLOW ); for( i = 0; i < count && val[i] != stop_value; i++ ) cvWriteReal( fs, 0, val[i] ); cvEndWriteStruct( fs ); } } int CvTest::read_params( CvFileStorage* fs ) { int code = 0; if(fs == NULL) return code; if( ts->get_testing_mode() == CvTS::TIMING_MODE ) { timing_param_names = find_param( fs, "timing_params" ); if( CV_NODE_IS_SEQ(timing_param_names->tag) ) { CvSeq* seq = timing_param_names->data.seq; CvSeqReader reader; cvStartReadSeq( seq, &reader ); int i; timing_param_count = seq->total; timing_param_seqs = (const CvFileNode**)malloc( timing_param_count*sizeof(timing_param_seqs[0])); timing_param_idxs = (int*)malloc( timing_param_count*sizeof(timing_param_idxs[0])); timing_param_current = (const CvFileNode**)malloc( timing_param_count*sizeof(timing_param_current[0])); test_case_count = 1; for( i = 0; i < timing_param_count; i++ ) { CvFileNode* param_name = (CvFileNode*)(reader.ptr); if( !CV_NODE_IS_STRING(param_name->tag) ) { ts->printf( CvTS::LOG, "ERROR: name of timing parameter #%d is not a string\n", i ); code = -1; break; } timing_param_idxs[i] = 0; timing_param_current[i] = 0; timing_param_seqs[i] = find_param( fs, param_name->data.str.ptr ); if( !timing_param_seqs[i] ) { ts->printf( CvTS::LOG, "ERROR: timing parameter %s is not found\n", param_name->data.str.ptr ); code = -1; break; } if( CV_NODE_IS_SEQ(timing_param_seqs[i]->tag) ) test_case_count *= timing_param_seqs[i]->data.seq->total; CV_NEXT_SEQ_ELEM( seq->elem_size, reader ); } if( i < timing_param_count ) timing_param_count = 0; } else { ts->printf( CvTS::LOG, "ERROR: \"timing_params\" is not found" ); code = -1; } } return code; } int CvTest::get_next_timing_param_tuple() { bool increment; int i; if( timing_param_count <= 0 || !timing_param_names || !timing_param_seqs ) return -1; increment = timing_param_current[0] != 0; // if already have some valid test tuple, move to the next for( i = 0; i < timing_param_count; i++ ) { const CvFileNode* node = timing_param_seqs[i]; int total = CV_NODE_IS_SEQ(node->tag) ? node->data.seq->total : 1; int new_idx = timing_param_idxs[i]; if( !timing_param_current[i] ) timing_param_idxs[i] = new_idx = 0; else if( increment ) { new_idx++; if( new_idx >= total ) new_idx = 0; else if( total > 1 ) increment = false; } if( !timing_param_current[i] || new_idx != timing_param_idxs[i] ) { if( CV_NODE_IS_SEQ(node->tag) ) timing_param_current[i] = (CvFileNode*)cvGetSeqElem( node->data.seq, new_idx ); else timing_param_current[i] = node; timing_param_idxs[i] = new_idx; } } return !increment; // return 0 in case of overflow (i.e. if there is no more test cases) } const CvFileNode* CvTest::find_timing_param( const char* paramname ) { if( timing_param_names ) { int i; CvSeqReader reader; cvStartReadSeq( timing_param_names->data.seq, &reader, 0 ); for( i = 0; i < timing_param_count; i++ ) { const char* ptr = ((const CvFileNode*)(reader.ptr))->data.str.ptr; if( ptr[0] == paramname[0] && strcmp(ptr, paramname) == 0 ) return timing_param_current[i]; CV_NEXT_SEQ_ELEM( reader.seq->elem_size, reader ); } } return 0; } int CvTest::write_defaults(CvTS* _ts) { ts = _ts; hdr_state = 0; write_default_params( ts->get_file_storage() ); if( hdr_state ) cvEndWriteStruct( ts->get_file_storage() ); return 0; } int CvTest::write_default_params( CvFileStorage* fs ) { if( ts->get_testing_mode() == CvTS::TIMING_MODE ) write_string_list( fs, "timing_params", default_timing_param_names, timing_param_count ); return 0; } bool CvTest::can_do_fast_forward() { return true; } int CvTest::get_support_testing_modes() { return support_testing_modes; } void CvTest::safe_run( int start_from ) { if(ts->is_debug_mode()) run( start_from ); else { try { #if !defined WIN32 && !defined _WIN32 int _code = setjmp( cv_ts_jmp_mark ); if( !_code ) run( start_from ); else throw _code; #else run( start_from ); #endif } catch (const cv::Exception& exc) { const char* errorStr = cvErrorStr(exc.code); char buf[1 << 16]; sprintf( buf, "OpenCV Error: %s (%s) in %s, file %s, line %d", errorStr, exc.err.c_str(), exc.func.size() > 0 ? exc.func.c_str() : "unknown function", exc.file.c_str(), exc.line ); ts->printf(CvTS::LOG, "%s\n", buf); ts->set_failed_test_info( CvTS::FAIL_ERROR_IN_CALLED_FUNC ); } catch (...) { ts->set_failed_test_info( CvTS::FAIL_EXCEPTION ); } } } void CvTest::run( int start_from ) { int i, test_case_idx, count = get_test_case_count(); int64 t_start = cvGetTickCount(); double freq = cv::getTickFrequency(); bool ff = can_do_fast_forward(); int progress = 0, code; std::vector v_cpe, v_time; int64 t1 = t_start; for( test_case_idx = ff && start_from >= 0 ? start_from : 0; count < 0 || test_case_idx < count; test_case_idx++ ) { ts->update_context( this, test_case_idx, ff ); progress = update_progress( progress, test_case_idx, count, (double)(t1 - t_start)/(freq*1000) ); int64 t00 = 0, t0 = 0, t2 = 0, t3 = 0; double t_acc = 0, t_cpu_acc = 0; if( ts->get_testing_mode() == CvTS::TIMING_MODE ) { const int iterations = 20; code = prepare_test_case( test_case_idx ); if( code < 0 || ts->get_err_code() < 0 ) return; if( code == 0 ) continue; v_cpe.resize(0); v_time.resize(0); for( i = 0; i < iterations; i++ ) { for(;;) { t0 = cv::getTickCount(); t2 = cv::getCPUTickCount(); run_func(); t3 = cv::getCPUTickCount(); t1 = cv::getTickCount(); if( ts->get_err_code() < 0 ) return; if( t3 - t2 > 0 && t1 - t0 > 1 ) break; } if( i == 0 ) t00 = t0; v_cpe.push_back((double)(t3 - t2)); v_time.push_back((double)(t1 - t0)); if( i >= 5 && t1 - t00 > freq*5 ) break; } std::sort(v_cpe.begin(), v_cpe.end()); std::sort(v_time.begin(), v_time.end()); t_cpu_acc = v_cpe[i/2]; t_acc = v_time[i/2]; print_time( test_case_idx, t_acc, t_cpu_acc ); } else { code = prepare_test_case( test_case_idx ); if( code < 0 || ts->get_err_code() < 0 ) return; if( code == 0 ) continue; run_func(); if( ts->get_err_code() < 0 ) return; if( validate_test_results( test_case_idx ) < 0 || ts->get_err_code() < 0 ) return; } } } void CvTest::run_func() { assert(0); } int CvTest::get_test_case_count() { return test_case_count; } int CvTest::prepare_test_case( int ) { return 0; } int CvTest::validate_test_results( int ) { return 0; } void CvTest::print_time( int /*test_case_idx*/, double /*time_usecs*/, double /*time_cpu_clocks*/ ) { } int CvTest::update_progress( int progress, int test_case_idx, int count, double dt ) { int width = 60 - (int)strlen(get_name()); if( count > 0 ) { int t = cvRound( ((double)test_case_idx * width)/count ); if( t > progress ) { ts->printf( CvTS::CONSOLE, "." ); progress = t; } } else if( cvRound(dt) > progress ) { ts->printf( CvTS::CONSOLE, "." ); progress = cvRound(dt); } return progress; } CvBadArgTest::CvBadArgTest( const char* _test_name, const char* _test_funcs, const char* _test_descr ) : CvTest( _test_name, _test_funcs, _test_descr ) { progress = -1; test_case_idx = -1; freq = cv::getTickFrequency(); } CvBadArgTest::~CvBadArgTest() { } int CvBadArgTest::run_test_case( int expected_code, const char* descr ) { double new_t = (double)cv::getTickCount(), dt; if( test_case_idx < 0 ) { test_case_idx = 0; progress = 0; dt = 0; } else { dt = (new_t - t)/(freq*1000); t = new_t; } progress = update_progress(progress, test_case_idx, 0, dt); int errcount = 0; bool thrown = false; if(!descr) descr = ""; try { run_func(); } catch(const cv::Exception& e) { thrown = true; if( e.code != expected_code ) { ts->printf(CvTS::LOG, "%s (test case #%d): the error code %d is different from the expected %d\n", descr, test_case_idx, e.code, expected_code); errcount = 1; } } catch(...) { thrown = true; ts->printf(CvTS::LOG, "%s (test case #%d): unknown exception was thrown (the function has likely crashed)\n", descr, test_case_idx); errcount = 1; } if(!thrown) { ts->printf(CvTS::LOG, "%s (test case #%d): no expected exception was thrown\n", descr, test_case_idx); errcount = 1; } test_case_idx++; return errcount; } /*****************************************************************************************\ * Base Class for Test System * \*****************************************************************************************/ /******************************** Constructors/Destructors ******************************/ CvTS::CvTS(const char* _module_name) { module_name = _module_name; start_time = 0; version = CV_TS_VERSION; memory_manager = 0; /* memory_manager = new CvTestMemoryManager(this); cvSetMemoryManager( CvTestMemoryManager::alloc_proxy, CvTestMemoryManager::free_proxy, memory_manager );*/ ostrm_suffixes[SUMMARY_IDX] = ".sum"; ostrm_suffixes[LOG_IDX] = ".log"; ostrm_suffixes[CSV_IDX] = ".csv"; ostrm_suffixes[CONSOLE_IDX] = 0; ostrm_base_name = 0; memset( output_streams, 0, sizeof(output_streams) ); memset( ¶ms, 0, sizeof(params) ); selected_tests = new CvTestPtrVec(); failed_tests = new CvTestInfoVec(); written_params = new CvTestPtrVec(); clear(); } void CvTS::clear() { int i; CvTest* test; for( test = get_first_test(); test != 0; test = test->get_next() ) test->clear(); for( i = 0; i <= CONSOLE_IDX; i++ ) { if( i == LOG_IDX ) fflush( stderr ); else if( i == CONSOLE_IDX ) fflush( stdout ); if( i < CONSOLE_IDX && output_streams[i].f ) { fclose( output_streams[i].f ); output_streams[i].f = 0; } if( i == LOG_IDX && output_streams[i].default_handle > 0 ) { dup2( output_streams[i].default_handle, 2 ); output_streams[i].default_handle = 0; } output_streams[i].enable = 1; } cvReleaseFileStorage( &fs ); selected_tests->clear(); failed_tests->clear(); if( ostrm_base_name ) { free( ostrm_base_name ); ostrm_base_name = 0; } params.rng_seed = 0; params.debug_mode = -1; params.print_only_failed = 0; params.skip_header = 0; params.test_mode = CORRECTNESS_CHECK_MODE; params.timing_mode = MIN_TIME; params.use_optimized = -1; params.color_terminal = 1; if( memory_manager ) memory_manager->clear_and_check(); } CvTS::~CvTS() { clear(); set_data_path(0); if( written_params ) { for( int i = 0; i < written_params->size(); i++ ) free( written_params->at(i) ); delete written_params; } delete selected_tests; delete failed_tests; } const char* CvTS::str_from_code( int code ) { switch( code ) { case OK: return "Ok"; case FAIL_GENERIC: return "Generic/Unknown"; case FAIL_MISSING_TEST_DATA: return "No test data"; case FAIL_INVALID_TEST_DATA: return "Invalid test data"; case FAIL_ERROR_IN_CALLED_FUNC: return "cvError invoked"; case FAIL_EXCEPTION: return "Hardware/OS exception"; case FAIL_MEMORY_EXCEPTION: return "Invalid memory access"; case FAIL_ARITHM_EXCEPTION: return "Arithmetic exception"; case FAIL_MEMORY_CORRUPTION_BEGIN: return "Corrupted memblock (beginning)"; case FAIL_MEMORY_CORRUPTION_END: return "Corrupted memblock (end)"; case FAIL_MEMORY_LEAK: return "Memory leak"; case FAIL_INVALID_OUTPUT: return "Invalid function output"; case FAIL_MISMATCH: return "Unexpected output"; case FAIL_BAD_ACCURACY: return "Bad accuracy"; case FAIL_HANG: return "Infinite loop(?)"; case FAIL_BAD_ARG_CHECK: return "Incorrect handling of bad arguments"; default: return "Generic/Unknown"; } } /************************************** Running tests **********************************/ void CvTS::make_output_stream_base_name( const char* config_name ) { int k, len = (int)strlen( config_name ); if( ostrm_base_name ) free( ostrm_base_name ); for( k = len-1; k >= 0; k-- ) { char c = config_name[k]; if( c == '.' || c == '/' || c == '\\' || c == ':' ) break; } if( k > 0 && config_name[k] == '.' ) len = k; ostrm_base_name = (char*)malloc( len + 1 ); memcpy( ostrm_base_name, config_name, len ); ostrm_base_name[len] = '\0'; } void CvTS::set_handlers( bool on ) { if( on ) { cvSetErrMode( CV_ErrModeParent ); cvRedirectError( cvStdErrReport ); #if defined WIN32 || defined _WIN32 #ifdef _MSC_VER _set_se_translator( cv_seh_translator ); #endif #else for( int i = 0; cv_ts_sig_id[i] >= 0; i++ ) signal( cv_ts_sig_id[i], cv_signal_handler ); #endif } else { cvSetErrMode( CV_ErrModeLeaf ); cvRedirectError( cvGuiBoxReport ); #if defined WIN32 || defined _WIN32 #ifdef _MSC_VER _set_se_translator( 0 ); #endif #else for( int i = 0; cv_ts_sig_id[i] >= 0; i++ ) signal( cv_ts_sig_id[i], SIG_DFL ); #endif } } void CvTS::set_data_path( const char* data_path ) { if( data_path == params.data_path ) return; if( params.data_path ) delete[] params.data_path; if( data_path ) { int size = (int)strlen(data_path)+1; bool append_slash = data_path[size-1] != '/' && data_path[size-1] != '\\'; params.data_path = new char[size+1]; memcpy( params.data_path, data_path, size ); if( append_slash ) strcat( params.data_path, "/" ); } } typedef struct CvTsParamVal { const char* fullname; const void* val; } CvTsParamVal; int CvTS::find_written_param( CvTest* test, const char* paramname, int valtype, const void* val ) { const char* testname = test->get_name(); bool add_to_list = test->get_func_list()[0] == '\0'; char buffer[256]; int paramname_len = (int)strlen(paramname); int paramval_len = valtype == CV_NODE_INT ? (int)sizeof(int) : valtype == CV_NODE_REAL ? (int)sizeof(double) : -1; const char* name = CvTest::get_parent_name( testname, buffer ); if( !fs ) return -1; if( paramval_len < 0 ) { assert(0); // unsupported parameter type return -1; } while( name ) { int i, len = (int)strlen(buffer); buffer[len] = '.'; memcpy( buffer + len + 1, paramname, paramname_len + 1 ); for( i = 0; i < written_params->size(); i++ ) { CvTsParamVal* param = (CvTsParamVal*)written_params->at(i); if( strcmp( param->fullname, buffer ) == 0 ) { if( (paramval_len > 0 && memcmp( param->val, val, paramval_len ) == 0) || (paramval_len < 0 && strcmp( (const char*)param->val, (const char*)val ) == 0) ) return 1; break; } } if( i < written_params->size() ) break; buffer[len] = '\0'; name = CvTest::get_parent_name( buffer, buffer ); } if( add_to_list ) { int bufsize, fullname_len = (int)strlen(testname) + paramname_len + 2; CvTsParamVal* param; if( paramval_len < 0 ) paramval_len = (int)strlen((const char*)val) + 1; bufsize = sizeof(*param) + fullname_len + paramval_len; param = (CvTsParamVal*)malloc(bufsize); param->fullname = (const char*)(param + 1); param->val = param->fullname + fullname_len; sprintf( (char*)param->fullname, "%s.%s", testname, paramname ); memcpy( (void*)param->val, val, paramval_len ); written_params->push( param ); } return 0; } #ifndef MAX_PATH #define MAX_PATH 1024 #endif static int CV_CDECL cmp_test_names( const void* a, const void* b ) { return strcmp( (*(const CvTest**)a)->get_name(), (*(const CvTest**)b)->get_name() ); } int CvTS::run( int argc, char** argv, const char** blacklist ) { time( &start_time ); int i, write_params = 0; int list_tests = 0; CvTestPtrVec all_tests; CvTest* test; // 0. reset all the parameters, reorder tests clear(); /*#if defined WIN32 || defined _WIN32 cv::setBreakOnError(true); #endif*/ for( test = get_first_test(), i = 0; test != 0; test = test->get_next(), i++ ) all_tests.push(test); if( all_tests.size() > 0 && all_tests.data() ) qsort( all_tests.data(), all_tests.size(), sizeof(CvTest*), cmp_test_names ); // 1. parse command line options for( i = 1; i < argc; i++ ) { if( strcmp( argv[i], "-h" ) == 0 || strcmp( argv[i], "--help" ) == 0 ) { print_help(); return 0; } else if( strcmp( argv[i], "-f" ) == 0 ) config_name = argv[++i]; else if( strcmp( argv[i], "-w" ) == 0 ) write_params = 1; else if( strcmp( argv[i], "-t" ) == 0 ) params.test_mode = TIMING_MODE; else if( strcmp( argv[i], "-O0" ) == 0 || strcmp( argv[i], "-O1" ) == 0 ) params.use_optimized = argv[i][2] - '0'; else if( strcmp( argv[i], "-l" ) == 0 ) list_tests = 1; else if( strcmp( argv[i], "-d" ) == 0 ) set_data_path(argv[++i]); else if( strcmp( argv[i], "-nc" ) == 0 ) params.color_terminal = 0; else if( strcmp( argv[i], "-r" ) == 0 ) params.debug_mode = 0; else if( strcmp( argv[i], "-tn" ) == 0 ) { params.test_filter_pattern = argv[++i]; params.test_filter_mode = CHOOSE_TESTS; } else if( strcmp( argv[i], "-seed" ) == 0 ) { params.rng_seed = read_seed(argv[++i]); if( params.rng_seed == 0 ) fprintf(stderr, "Invalid or zero RNG seed. Will use the seed from the config file or default one\n"); } } // this is the fallback for the current OpenCV autotools setup if( !params.data_path || !params.data_path[0] ) { char* datapath_dir = getenv("OPENCV_TEST_DATA_PATH"); char buf[1024]; if( datapath_dir ) { sprintf( buf, "%s/%s", datapath_dir, module_name ? module_name : "" ); printf( LOG + SUMMARY, "Data Path = %s\n", buf); set_data_path(buf); } } if( write_params ) { if( !config_name ) { printf( LOG, "ERROR: output config name is not specified\n" ); return -1; } fs = cvOpenFileStorage( config_name, 0, CV_STORAGE_WRITE ); if( !fs ) { printf( LOG, "ERROR: could not open config file %s\n", config_name ); return -1; } cvWriteComment( fs, CV_TS_VERSION " config file", 0 ); cvStartWriteStruct( fs, "common", CV_NODE_MAP ); write_default_params( fs ); cvEndWriteStruct( fs ); for( i = 0; i < all_tests.size(); i++ ) { test = (CvTest*)all_tests[i]; if( !(test->get_support_testing_modes() & get_testing_mode()) ) continue; test->write_defaults( this ); test->clear(); } cvReleaseFileStorage( &fs ); return 0; } if( !config_name ) printf( LOG, "WARNING: config name is not specified, using default parameters\n" ); else { // 2. read common parameters of test system fs = cvOpenFileStorage( config_name, 0, CV_STORAGE_READ ); if( !fs ) { printf( LOG, "ERROR: could not open config file %s", config_name ); return -1; } } if( params.test_mode == CORRECTNESS_CHECK_MODE || fs ) { // in the case of algorithmic tests we always run read_params, // even if there is no config file if( read_params(fs) < 0 ) return -1; } if( !ostrm_base_name ) make_output_stream_base_name( config_name ? config_name : argv[0] ); ostream_testname_mask = -1; // disable printing test names at initial stage // 3. open file streams for( i = 0; i < CONSOLE_IDX; i++ ) { char filename[MAX_PATH]; sprintf( filename, "%s%s", ostrm_base_name, ostrm_suffixes[i] ); output_streams[i].f = fopen( filename, "wt" ); if( !output_streams[i].f ) { printf( LOG, "ERROR: could not open %s\n", filename ); return -1; } if( i == LOG_IDX ) { // redirect stderr to log file fflush( stderr ); output_streams[i].default_handle = dup(2); dup2( fileno(output_streams[i].f), 2 ); } } int filter_state = 0; // 4. traverse through the list of all registered tests. // Initialize the selected tests and put them into the separate sequence for( i = 0; i < all_tests.size(); i++ ) { test = (CvTest*)all_tests[i]; if( !(test->get_support_testing_modes() & get_testing_mode()) ) continue; if( strcmp( test->get_func_list(), "" ) != 0 && filter(test, filter_state, blacklist) ) { if( test->init(this) >= 0 ) { selected_tests->push( test ); if( list_tests ) ::printf( "%s\n", test->get_name() ); } else printf( LOG, "WARNING: an error occured during test %s initialization\n", test->get_name() ); } } if( list_tests ) { clear(); return 0; } // 5. setup all the neccessary handlers and print header set_handlers( !params.debug_mode ); if( params.use_optimized == 0 ) cvUseOptimized(0); if( !params.skip_header ) print_summary_header( SUMMARY + LOG + CONSOLE + CSV ); rng = params.rng_seed; update_context( 0, -1, true ); // 6. run all the tests for( i = 0; i < selected_tests->size(); i++ ) { CvTest* test = (CvTest*)selected_tests->at(i); int code; CvTestInfo temp; if( memory_manager ) memory_manager->start_tracking(); update_context( test, -1, true ); current_test_info.rng_seed0 = current_test_info.rng_seed; ostream_testname_mask = 0; // reset "test name was printed" flags logbuf = std::string(); if( output_streams[LOG_IDX].f ) fflush( output_streams[LOG_IDX].f ); temp = current_test_info; test->safe_run(0); if( get_err_code() >= 0 ) { update_context( test, -1, false ); current_test_info.rng_seed = temp.rng_seed; current_test_info.base_alloc_index = temp.base_alloc_index; } test->clear(); if( memory_manager ) memory_manager->stop_tracking_and_check(); code = get_err_code(); if( code >= 0 ) { if( !params.print_only_failed ) { printf( SUMMARY + CONSOLE, "\t" ); set_color( CV_TS_GREEN ); printf( SUMMARY + CONSOLE, "Ok\n" ); set_color( CV_TS_NORMAL ); } } else { printf( SUMMARY + CONSOLE, "\t" ); set_color( CV_TS_RED ); printf( SUMMARY + CONSOLE, "FAIL(%s)\n", str_from_code(code) ); set_color( CV_TS_NORMAL ); printf( LOG, "context: test case = %d, seed = %08x%08x\n", current_test_info.test_case_idx, (unsigned)(current_test_info.rng_seed>>32), (unsigned)(current_test_info.rng_seed)); if(logbuf.size() > 0) { printf( SUMMARY + CONSOLE, ">>>\n%s\n", logbuf.c_str()); } failed_tests->push(current_test_info); if( params.rerun_immediately ) break; } } ostream_testname_mask = -1; print_summary_tailer( SUMMARY + CONSOLE + LOG ); if( !params.debug_mode && (params.rerun_failed || params.rerun_immediately) ) { set_handlers(0); update_context( 0, -1, true ); for( i = 0; i < failed_tests->size(); i++ ) { CvTestInfo info = failed_tests->at(i); if( (info.code == FAIL_MEMORY_CORRUPTION_BEGIN || info.code == FAIL_MEMORY_CORRUPTION_END || info.code == FAIL_MEMORY_LEAK) && memory_manager ) memory_manager->start_tracking( info.alloc_index - info.base_alloc_index + memory_manager->get_alloc_index() ); rng = info.rng_seed; test->safe_run( info.test_case_idx ); } } int nfailed = failed_tests ? (int)failed_tests->size() : 0; clear(); return nfailed; } void CvTS::print_help() { ::printf( "Usage: [{-h|--help}][-l] [-r] [-w] [-t] [-f ] [-d ] [-O{0|1}] [-tn ]\n\n" "-d - specify the test data path\n" "-f - use parameters from the provided XML/YAML config file\n" " instead of the default parameters\n" "-h or --help - print this help information\n" "-l - list all the registered tests or subset of the tests,\n" " selected in the config file, and exit\n" "-tn - only run a specific test\n" "-nc - do not use colors in the console output\n" "-O{0|1} - disable/enable on-fly detection of IPP and other\n" " supported optimized libs. It's enabled by default\n" "-r - continue running tests after OS/Hardware exception occured\n" "-t - switch to the performance testing mode instead of\n" " the default algorithmic/correctness testing mode\n" "-w - write default parameters of the algorithmic or\n" " performance (when -t is passed) tests to the specifed\n" " config file (see -f) and exit\n\n" //"Test data path and config file can also be specified by the environment variables 'config' and 'datapath'.\n\n" ); } #if defined WIN32 || defined _WIN32 const char* default_data_path = "../tests/cv/testdata/"; #else const char* default_data_path = "../../../../tests/cv/testdata/"; #endif int CvTS::read_params( CvFileStorage* fs ) { CvFileNode* node = fs ? cvGetFileNodeByName( fs, 0, "common" ) : 0; if(params.debug_mode < 0) params.debug_mode = cvReadIntByName( fs, node, "debug_mode", 1 ) != 0; params.skip_header = cvReadIntByName( fs, node, "skip_header", 0 ) != 0; params.print_only_failed = cvReadIntByName( fs, node, "print_only_failed", 0 ) != 0; params.rerun_failed = cvReadIntByName( fs, node, "rerun_failed", 0 ) != 0; params.rerun_immediately = cvReadIntByName( fs, node, "rerun_immediately", 0 ) != 0; const char* str = cvReadStringByName( fs, node, "filter_mode", "tests" ); params.test_filter_mode = strcmp( str, "functions" ) == 0 ? CHOOSE_FUNCTIONS : CHOOSE_TESTS; str = cvReadStringByName( fs, node, "test_mode", params.test_mode == TIMING_MODE ? "timing" : "correctness" ); params.test_mode = strcmp( str, "timing" ) == 0 || strcmp( str, "performance" ) == 0 ? TIMING_MODE : CORRECTNESS_CHECK_MODE; str = cvReadStringByName( fs, node, "timing_mode", params.timing_mode == AVG_TIME ? "avg" : "min" ); params.timing_mode = strcmp( str, "average" ) == 0 || strcmp( str, "avg" ) == 0 ? AVG_TIME : MIN_TIME; params.test_filter_pattern = params.test_filter_pattern != 0 && strlen(params.test_filter_pattern) > 0 ? params.test_filter_pattern : cvReadStringByName( fs, node, params.test_filter_mode == CHOOSE_FUNCTIONS ? "functions" : "tests", "" ); params.resource_path = cvReadStringByName( fs, node, "." ); if( params.use_optimized < 0 ) params.use_optimized = cvReadIntByName( fs, node, "use_optimized", -1 ); if( !params.data_path || !params.data_path[0] ) { const char* data_path = cvReadStringByName( fs, node, "data_path", default_data_path ); set_data_path(data_path); } params.test_case_count_scale = cvReadRealByName( fs, node, "test_case_count_scale", 1. ); if( params.test_case_count_scale <= 0 ) params.test_case_count_scale = 1.; str = cvReadStringByName( fs, node, "seed", 0 ); if( str && params.rng_seed == 0 ) params.rng_seed = read_seed(str); if( params.rng_seed == 0 ) params.rng_seed = cvGetTickCount(); str = cvReadStringByName( fs, node, "output_file_base_name", 0 ); if( str ) make_output_stream_base_name( str ); return 0; } void CvTS::write_default_params( CvFileStorage* fs ) { read_params(0); // fill parameters with default values cvWriteInt( fs, "debug_mode", params.debug_mode ); cvWriteInt( fs, "skip_header", params.skip_header ); cvWriteInt( fs, "print_only_failed", params.print_only_failed ); cvWriteInt( fs, "rerun_failed", params.rerun_failed ); cvWriteInt( fs, "rerun_immediately", params.rerun_immediately ); cvWriteString( fs, "filter_mode", params.test_filter_mode == CHOOSE_FUNCTIONS ? "functions" : "tests" ); cvWriteString( fs, "test_mode", params.test_mode == TIMING_MODE ? "timing" : "correctness" ); cvWriteString( fs, "data_path", params.data_path ? params.data_path : default_data_path, 1 ); if( params.test_mode == TIMING_MODE ) cvWriteString( fs, "timing_mode", params.timing_mode == AVG_TIME ? "avg" : "min" ); // test_filter, seed & output_file_base_name are not written } void CvTS::enable_output_streams( int stream_mask, int value ) { for( int i = 0; i < MAX_IDX; i++ ) if( stream_mask & (1 << i) ) output_streams[i].enable = value != 0; } void CvTS::update_context( CvTest* test, int test_case_idx, bool update_ts_context ) { current_test_info.test = test; current_test_info.test_case_idx = test_case_idx; current_test_info.alloc_index = 0; current_test_info.code = 0; cvSetErrStatus( CV_StsOk ); if( update_ts_context ) { current_test_info.rng_seed = rng; current_test_info.base_alloc_index = memory_manager ? memory_manager->get_alloc_index() : 0; } } void CvTS::set_failed_test_info( int fail_code, int alloc_index ) { if( fail_code == FAIL_MEMORY_CORRUPTION_BEGIN || fail_code == FAIL_MEMORY_CORRUPTION_END || current_test_info.code >= 0 ) { current_test_info.code = fail_code; current_test_info.alloc_index = alloc_index; } } const char* CvTS::get_libs_info( const char** addon_modules ) { const char* all_info = 0; cvGetModuleInfo( 0, &all_info, addon_modules ); return all_info; } void CvTS::print_summary_header( int streams ) { char csv_header[256], *ptr = csv_header; int i; printf( streams, "Engine: %s\n", version ); time_t t1; time( &t1 ); struct tm *t2 = localtime( &t1 ); char buf[1024]; strftime( buf, sizeof(buf)-1, "%c", t2 ); printf( streams, "Execution Date & Time: %s\n", buf ); printf( streams, "Config File: %s\n", config_name ); const char* plugins = 0; const char* lib_verinfo = get_libs_info( &plugins ); printf( streams, "Tested Libraries: %s\n", lib_verinfo ); printf( streams, "Optimized Low-level Plugin\'s: %s\n", plugins ); printf( streams, "=================================================\n"); sprintf( ptr, "funcName,dataType,channels,size," ); ptr += strlen(ptr); for( i = 0; i < CvTest::TIMING_EXTRA_PARAMS; i++ ) { sprintf( ptr, "param%d,", i ); ptr += strlen(ptr); } sprintf( ptr, "CPE,Time(uSecs)" ); printf( CSV, "%s\n", csv_header ); } void CvTS::print_summary_tailer( int streams ) { printf( streams, "=================================================\n"); if( selected_tests && failed_tests ) { time_t end_time; time( &end_time ); double total_time = difftime( end_time, start_time ); printf( streams, "Summary: %d out of %d tests failed\n", failed_tests->size(), selected_tests->size() ); int minutes = cvFloor(total_time/60.); int seconds = cvRound(total_time - minutes*60); int hours = minutes / 60; minutes %= 60; printf( streams, "Running time: %02d:%02d:%02d\n", hours, minutes, seconds ); } } #if defined _MSC_VER && _MSC_VER < 1400 #undef vsnprintf #define vsnprintf _vsnprintf #endif void CvTS::vprintf( int streams, const char* fmt, va_list l ) { if( streams ) { char str[1 << 14]; vsnprintf( str, sizeof(str)-1, fmt, l ); for( int i = 0; i < MAX_IDX; i++ ) { if( (streams & (1 << i)) && output_streams[i].enable ) { FILE* f = i == CONSOLE_IDX ? stdout : i == LOG_IDX ? stderr : output_streams[i].f; if( f ) { if( i != CSV_IDX && !(ostream_testname_mask & (1 << i)) && current_test_info.test ) { fprintf( f, "-------------------------------------------------\n" ); if( i == CONSOLE_IDX || i == SUMMARY_IDX ) fprintf( f, "[%08x%08x]\n", (int)(current_test_info.rng_seed0 >> 32), (int)(current_test_info.rng_seed0)); fprintf( f, "%s: ", current_test_info.test->get_name() ); fflush( f ); ostream_testname_mask |= 1 << i; if( i == LOG_IDX ) logbuf = std::string(); } fputs( str, f ); if( i == LOG_IDX ) logbuf += std::string(str); if( i == CONSOLE_IDX ) fflush(f); } } } } } void CvTS::printf( int streams, const char* fmt, ... ) { if( streams ) { va_list l; va_start( l, fmt ); vprintf( streams, fmt, l ); va_end( l ); } } void CvTS::set_color(int color) { if( params.color_terminal ) change_color(color); } static char* cv_strnstr( const char* str, int len, const char* pattern, int pattern_len = -1, int whole_word = 1 ) { int i; if( len < 0 && pattern_len < 0 ) return (char*)strstr( str, pattern ); if( len < 0 ) len = (int)strlen( str ); if( pattern_len < 0 ) pattern_len = (int)strlen( pattern ); for( i = 0; i < len - pattern_len + 1; i++ ) { int j = i + pattern_len; if( str[i] == pattern[0] && memcmp( str + i, pattern, pattern_len ) == 0 && (!whole_word || ((i == 0 || (!isalnum(str[i-1]) && str[i-1] != '_')) && (j == len || (!isalnum(str[j]) && str[j] != '_'))))) return (char*)(str + i); } return 0; } int CvTS::filter( CvTest* test, int& filter_state, const char** blacklist ) { const char* pattern = params.test_filter_pattern; const char* test_name = test->get_name(); int inverse = 0; int greater_or_equal = 0; if( blacklist ) { for( ; *blacklist != 0; blacklist++ ) { if( strcmp( *blacklist, test_name ) == 0 ) return 0; } } if( pattern && pattern[0] == '!' ) { inverse = 1; pattern++; } if( pattern && pattern[0] == '>' ) { greater_or_equal = 1; pattern++; if( pattern[0] == '=' ) { greater_or_equal = 2; pattern++; } } if( !pattern || strcmp( pattern, "" ) == 0 || strcmp( pattern, "*" ) == 0 ) return 1 ^ inverse; if( params.test_filter_mode == CHOOSE_TESTS ) { int found = 0; while( pattern && *pattern ) { char *ptr, *endptr = (char*)strchr( pattern, ',' ); int len, have_wildcard; int t_name_len; if( endptr ) *endptr = '\0'; ptr = (char*)strchr( pattern, '*' ); if( ptr ) { len = (int)(ptr - pattern); have_wildcard = 1; } else { len = (int)strlen( pattern ); have_wildcard = 0; } t_name_len = (int)strlen( test_name ); found = (t_name_len == len || (have_wildcard && t_name_len > len)) && (len == 0 || memcmp( test_name, pattern, len ) == 0); if( endptr ) { *endptr = ','; pattern = endptr + 1; while( isspace(*pattern) ) pattern++; } if( found || !endptr ) break; } if( greater_or_equal == 0 ) return found ^ inverse; if( filter_state ) return inverse^1; if( !found ) return inverse; if( greater_or_equal == 1 ) { filter_state = 1; return inverse; } else { assert(filter_state == 2); filter_state = 1; return inverse ^ 1; } } else { assert( params.test_filter_mode == CHOOSE_FUNCTIONS ); int glob_len = (int)strlen( pattern ); const char* ptr = test->get_func_list(); const char *tmp_ptr; while( ptr && *ptr ) { const char* endptr = ptr - 1; const char* name_ptr; const char* name_first_match; int name_len; char c; do c = *++endptr; while( isspace(c) ); if( !c ) break; assert( isalpha(c) ); name_ptr = endptr; do c = *++endptr; while( isalnum(c) || c == '_' ); if( c == ':' ) // class { assert( endptr[1] == ':' ); endptr = endptr + 2; name_len = (int)(endptr - name_ptr); // find the first occurence of the class name // in pattern name_first_match = cv_strnstr( pattern, glob_len, name_ptr, name_len, 1 ); if( *endptr == '*' ) { if( name_first_match ) return 1 ^ inverse; } else { assert( *endptr == '{' ); // a list of methods if( !name_first_match ) { // skip all the methods, if there is no such a class name // in pattern endptr = strchr( endptr, '}' ); assert( endptr != 0 ); endptr--; } for( ;; ) { const char* method_name_ptr; int method_name_len; do c = *++endptr; while( isspace(c) ); if( c == '}' ) break; assert( isalpha(c) ); method_name_ptr = endptr; do c = *++endptr; while( isalnum(c) || c == '_' ); method_name_len = (int)(endptr - method_name_ptr); // search for class_name::* or // class_name::{...method_name...} tmp_ptr = name_first_match; do { const char* tmp_ptr2; tmp_ptr += name_len; if( *tmp_ptr == '*' ) return 1; assert( *tmp_ptr == '{' ); tmp_ptr2 = strchr( tmp_ptr, '}' ); assert( tmp_ptr2 ); if( cv_strnstr( tmp_ptr, (int)(tmp_ptr2 - tmp_ptr) + 1, method_name_ptr, method_name_len, 1 )) return 1 ^ inverse; tmp_ptr = cv_strnstr( tmp_ptr2, glob_len - (int)(tmp_ptr2 - pattern), name_ptr, name_len, 1 ); } while( tmp_ptr ); endptr--; do c = *++endptr; while( isspace(c) ); if( c != ',' ) endptr--; } } } else { assert( !c || isspace(c) || c == ',' ); name_len = (int)(endptr - name_ptr); tmp_ptr = pattern; for(;;) { const char *tmp_ptr2, *tmp_ptr3; tmp_ptr = cv_strnstr( tmp_ptr, glob_len - (int)(tmp_ptr - pattern), name_ptr, name_len, 1 ); if( !tmp_ptr ) break; // make sure it is not a method tmp_ptr2 = strchr( tmp_ptr, '}' ); if( !tmp_ptr2 ) return 1 ^ inverse; tmp_ptr3 = strchr( tmp_ptr, '{' ); if( tmp_ptr3 < tmp_ptr2 ) return 1 ^ inverse; tmp_ptr = tmp_ptr2 + 1; } endptr--; } do c = *++endptr; while( isspace(c) ); if( c == ',' ) endptr++; ptr = endptr; } return 0 ^ inverse; } } /* End of file. */