#include "precomp.hpp" #include #include #include "opencv2/imgproc/types_c.h" using namespace cv; namespace cvtest { const char* getTypeName( int type ) { static const char* type_names[] = { "8u", "8s", "16u", "16s", "32s", "32f", "64f", "ptr" }; return type_names[CV_MAT_DEPTH(type)]; } int typeByName( const char* name ) { int i; for( i = 0; i < CV_DEPTH_MAX; i++ ) if( strcmp(name, getTypeName(i)) == 0 ) return i; return -1; } string vec2str( const string& sep, const int* v, size_t nelems ) { char buf[32]; string result = ""; for( size_t i = 0; i < nelems; i++ ) { sprintf(buf, "%d", v[i]); result += string(buf); if( i < nelems - 1 ) result += sep; } return result; } Size randomSize(RNG& rng, double maxSizeLog) { double width_log = rng.uniform(0., maxSizeLog); double height_log = rng.uniform(0., maxSizeLog - width_log); if( (unsigned)rng % 2 != 0 ) std::swap(width_log, height_log); Size sz; sz.width = cvRound(exp(width_log)); sz.height = cvRound(exp(height_log)); return sz; } void randomSize(RNG& rng, int minDims, int maxDims, double maxSizeLog, vector& sz) { int i, dims = rng.uniform(minDims, maxDims+1); sz.resize(dims); for( i = 0; i < dims; i++ ) { double v = rng.uniform(0., maxSizeLog); maxSizeLog -= v; sz[i] = cvRound(exp(v)); } for( i = 0; i < dims; i++ ) { int j = rng.uniform(0, dims); int k = rng.uniform(0, dims); std::swap(sz[j], sz[k]); } } int randomType(RNG& rng, int typeMask, int minChannels, int maxChannels) { int channels = rng.uniform(minChannels, maxChannels+1); int depth = 0; CV_Assert((typeMask & _OutputArray::DEPTH_MASK_ALL_16F) != 0); for(;;) { depth = rng.uniform(CV_8U, CV_16F+1); if( ((1 << depth) & typeMask) != 0 ) break; } return CV_MAKETYPE(depth, channels); } double getMinVal(int depth) { depth = CV_MAT_DEPTH(depth); double val = depth == CV_8U ? 0 : depth == CV_8S ? SCHAR_MIN : depth == CV_16U ? 0 : depth == CV_16S ? SHRT_MIN : depth == CV_32S ? INT_MIN : depth == CV_32F ? -FLT_MAX : depth == CV_64F ? -DBL_MAX : -1; CV_Assert(val != -1); return val; } double getMaxVal(int depth) { depth = CV_MAT_DEPTH(depth); double val = depth == CV_8U ? UCHAR_MAX : depth == CV_8S ? SCHAR_MAX : depth == CV_16U ? USHRT_MAX : depth == CV_16S ? SHRT_MAX : depth == CV_32S ? INT_MAX : depth == CV_32F ? FLT_MAX : depth == CV_64F ? DBL_MAX : -1; CV_Assert(val != -1); return val; } Mat randomMat(RNG& rng, Size size, int type, double minVal, double maxVal, bool useRoi) { Size size0 = size; if( useRoi ) { size0.width += std::max(rng.uniform(0, 10) - 5, 0); size0.height += std::max(rng.uniform(0, 10) - 5, 0); } Mat m(size0, type); rng.fill(m, RNG::UNIFORM, minVal, maxVal); if( size0 == size ) return m; return m(Rect((size0.width-size.width)/2, (size0.height-size.height)/2, size.width, size.height)); } Mat randomMat(RNG& rng, const vector& size, int type, double minVal, double maxVal, bool useRoi) { int i, dims = (int)size.size(); vector size0(dims); vector r(dims); bool eqsize = true; for( i = 0; i < dims; i++ ) { size0[i] = size[i]; r[i] = Range::all(); if( useRoi ) { size0[i] += std::max(rng.uniform(0, 5) - 2, 0); r[i] = Range((size0[i] - size[i])/2, (size0[i] - size[i])/2 + size[i]); } eqsize = eqsize && size[i] == size0[i]; } Mat m(dims, &size0[0], type); rng.fill(m, RNG::UNIFORM, minVal, maxVal); if( eqsize ) return m; return m(&r[0]); } void add(const Mat& _a, double alpha, const Mat& _b, double beta, Scalar gamma, Mat& c, int ctype, bool calcAbs) { Mat a = _a, b = _b; if( a.empty() || alpha == 0 ) { // both alpha and beta can be 0, but at least one of a and b must be non-empty array, // otherwise we do not know the size of the output (and may be type of the output, when ctype<0) CV_Assert( !a.empty() || !b.empty() ); if( !b.empty() ) { a = b; alpha = beta; b = Mat(); beta = 0; } } if( b.empty() || beta == 0 ) { b = Mat(); beta = 0; } else CV_Assert(a.size == b.size); if( ctype < 0 ) ctype = a.depth(); ctype = CV_MAKETYPE(CV_MAT_DEPTH(ctype), a.channels()); c.create(a.dims, &a.size[0], ctype); const Mat *arrays[] = {&a, &b, &c, 0}; Mat planes[3], buf[3]; NAryMatIterator it(arrays, planes); size_t i, nplanes = it.nplanes; int cn=a.channels(); int total = (int)planes[0].total(), maxsize = std::min(12*12*std::max(12/cn, 1), total); CV_Assert(planes[0].rows == 1); buf[0].create(1, maxsize, CV_64FC(cn)); if(!b.empty()) buf[1].create(1, maxsize, CV_64FC(cn)); buf[2].create(1, maxsize, CV_64FC(cn)); scalarToRawData(gamma, buf[2].ptr(), CV_64FC(cn), (int)(maxsize*cn)); for( i = 0; i < nplanes; i++, ++it) { for( int j = 0; j < total; j += maxsize ) { int j2 = std::min(j + maxsize, total); Mat apart0 = planes[0].colRange(j, j2); Mat cpart0 = planes[2].colRange(j, j2); Mat apart = buf[0].colRange(0, j2 - j); apart0.convertTo(apart, apart.type(), alpha); size_t k, n = (j2 - j)*cn; double* aptr = apart.ptr(); const double* gptr = buf[2].ptr(); if( b.empty() ) { for( k = 0; k < n; k++ ) aptr[k] += gptr[k]; } else { Mat bpart0 = planes[1].colRange((int)j, (int)j2); Mat bpart = buf[1].colRange(0, (int)(j2 - j)); bpart0.convertTo(bpart, bpart.type(), beta); const double* bptr = bpart.ptr(); for( k = 0; k < n; k++ ) aptr[k] += bptr[k] + gptr[k]; } if( calcAbs ) for( k = 0; k < n; k++ ) aptr[k] = fabs(aptr[k]); apart.convertTo(cpart0, cpart0.type(), 1, 0); } } } template inline void convert_(const _Tp1* src, _Tp2* dst, size_t total, double alpha, double beta) { size_t i; if( alpha == 1 && beta == 0 ) for( i = 0; i < total; i++ ) dst[i] = saturate_cast<_Tp2>(src[i]); else if( beta == 0 ) for( i = 0; i < total; i++ ) dst[i] = saturate_cast<_Tp2>(src[i]*alpha); else for( i = 0; i < total; i++ ) dst[i] = saturate_cast<_Tp2>(src[i]*alpha + beta); } template inline void convertTo(const _Tp* src, void* dst, int dtype, size_t total, double alpha, double beta) { switch( CV_MAT_DEPTH(dtype) ) { case CV_8U: convert_(src, (uchar*)dst, total, alpha, beta); break; case CV_8S: convert_(src, (schar*)dst, total, alpha, beta); break; case CV_16U: convert_(src, (ushort*)dst, total, alpha, beta); break; case CV_16S: convert_(src, (short*)dst, total, alpha, beta); break; case CV_32S: convert_(src, (int*)dst, total, alpha, beta); break; case CV_32F: convert_(src, (float*)dst, total, alpha, beta); break; case CV_64F: convert_(src, (double*)dst, total, alpha, beta); break; default: CV_Assert(0); } } void convert(const Mat& src, cv::OutputArray _dst, int dtype, double alpha, double beta) { if (dtype < 0) dtype = _dst.depth(); dtype = CV_MAKETYPE(CV_MAT_DEPTH(dtype), src.channels()); _dst.create(src.dims, &src.size[0], dtype); Mat dst = _dst.getMat(); if( alpha == 0 ) { set( dst, Scalar::all(beta) ); return; } if( dtype == src.type() && alpha == 1 && beta == 0 ) { copy( src, dst ); return; } const Mat *arrays[]={&src, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].channels(); size_t i, nplanes = it.nplanes; for( i = 0; i < nplanes; i++, ++it) { const uchar* sptr = planes[0].ptr(); uchar* dptr = planes[1].ptr(); switch( src.depth() ) { case CV_8U: convertTo((const uchar*)sptr, dptr, dtype, total, alpha, beta); break; case CV_8S: convertTo((const schar*)sptr, dptr, dtype, total, alpha, beta); break; case CV_16U: convertTo((const ushort*)sptr, dptr, dtype, total, alpha, beta); break; case CV_16S: convertTo((const short*)sptr, dptr, dtype, total, alpha, beta); break; case CV_32S: convertTo((const int*)sptr, dptr, dtype, total, alpha, beta); break; case CV_32F: convertTo((const float*)sptr, dptr, dtype, total, alpha, beta); break; case CV_64F: convertTo((const double*)sptr, dptr, dtype, total, alpha, beta); break; } } } void copy(const Mat& src, Mat& dst, const Mat& mask, bool invertMask) { dst.create(src.dims, &src.size[0], src.type()); if(mask.empty()) { const Mat* arrays[] = {&src, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t i, nplanes = it.nplanes; size_t planeSize = planes[0].total()*src.elemSize(); for( i = 0; i < nplanes; i++, ++it ) memcpy(planes[1].ptr(), planes[0].ptr(), planeSize); return; } int mcn = mask.channels(); CV_Assert( src.size == mask.size && mask.depth() == CV_8U && (mcn == 1 || mcn == src.channels()) ); const Mat *arrays[]={&src, &dst, &mask, 0}; Mat planes[3]; NAryMatIterator it(arrays, planes); size_t j, k, elemSize = src.elemSize(), maskElemSize = mask.elemSize(), total = planes[0].total(); size_t i, nplanes = it.nplanes; size_t elemSize1 = src.elemSize1(); for( i = 0; i < nplanes; i++, ++it) { const uchar* sptr = planes[0].ptr(); uchar* dptr = planes[1].ptr(); const uchar* mptr = planes[2].ptr(); for( j = 0; j < total; j++, sptr += elemSize, dptr += elemSize, mptr += maskElemSize ) { if( mcn == 1) { if( (mptr[0] != 0) ^ invertMask ) for( k = 0; k < elemSize; k++ ) dptr[k] = sptr[k]; } else { for( int c = 0; c < mcn; c++ ) if( (mptr[c] != 0) ^ invertMask ) for( k = 0; k < elemSize1; k++ ) dptr[k + c * elemSize1] = sptr[k + c * elemSize1]; } } } } void set(Mat& dst, const Scalar& gamma, const Mat& mask) { double buf[12]; scalarToRawData(gamma, &buf, dst.type(), dst.channels()); const uchar* gptr = (const uchar*)&buf[0]; if(mask.empty()) { const Mat* arrays[] = {&dst, 0}; Mat plane; NAryMatIterator it(arrays, &plane); size_t i, nplanes = it.nplanes; size_t j, k, elemSize = dst.elemSize(), planeSize = plane.total()*elemSize; for( k = 1; k < elemSize; k++ ) if( gptr[k] != gptr[0] ) break; bool uniform = k >= elemSize; for( i = 0; i < nplanes; i++, ++it ) { uchar* dptr = plane.ptr(); if( uniform ) memset( dptr, gptr[0], planeSize ); else if( i == 0 ) { for( j = 0; j < planeSize; j += elemSize, dptr += elemSize ) for( k = 0; k < elemSize; k++ ) dptr[k] = gptr[k]; } else memcpy(dptr, dst.ptr(), planeSize); } return; } int cn = dst.channels(), mcn = mask.channels(); CV_Assert( dst.size == mask.size && (mcn == 1 || mcn == cn) ); const Mat *arrays[]={&dst, &mask, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t j, k, elemSize = dst.elemSize(), maskElemSize = mask.elemSize(), total = planes[0].total(); size_t i, nplanes = it.nplanes; size_t elemSize1 = dst.elemSize1(); for( i = 0; i < nplanes; i++, ++it) { uchar* dptr = planes[0].ptr(); const uchar* mptr = planes[1].ptr(); for( j = 0; j < total; j++, dptr += elemSize, mptr += maskElemSize ) { if( mcn == 1) { if( mptr[0] ) for( k = 0; k < elemSize; k++ ) dptr[k] = gptr[k]; } else { for( int c = 0; c < mcn; c++ ) if( mptr[c] ) for( k = 0; k < elemSize1; k++ ) dptr[k + c * elemSize1] = gptr[k + c * elemSize1]; } } } } void insert(const Mat& src, Mat& dst, int coi) { CV_Assert( dst.size == src.size && src.depth() == dst.depth() && 0 <= coi && coi < dst.channels() ); const Mat* arrays[] = {&src, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t i, nplanes = it.nplanes; size_t j, k, size0 = src.elemSize(), size1 = dst.elemSize(), total = planes[0].total(); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr = planes[0].ptr(); uchar* dptr = planes[1].ptr() + coi*size0; for( j = 0; j < total; j++, sptr += size0, dptr += size1 ) { for( k = 0; k < size0; k++ ) dptr[k] = sptr[k]; } } } void extract(const Mat& src, Mat& dst, int coi) { dst.create( src.dims, &src.size[0], src.depth() ); CV_Assert( 0 <= coi && coi < src.channels() ); const Mat* arrays[] = {&src, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t i, nplanes = it.nplanes; size_t j, k, size0 = src.elemSize(), size1 = dst.elemSize(), total = planes[0].total(); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr = planes[0].ptr() + coi*size1; uchar* dptr = planes[1].ptr(); for( j = 0; j < total; j++, sptr += size0, dptr += size1 ) { for( k = 0; k < size1; k++ ) dptr[k] = sptr[k]; } } } void transpose(const Mat& src, Mat& dst) { CV_Assert(src.data != dst.data && "Inplace is not support in cvtest::transpose"); CV_Assert(src.dims == 2); dst.create(src.cols, src.rows, src.type()); int i, j, k, esz = (int)src.elemSize(); for( i = 0; i < dst.rows; i++ ) { const uchar* sptr = src.ptr(0) + i*esz; uchar* dptr = dst.ptr(i); for( j = 0; j < dst.cols; j++, sptr += src.step[0], dptr += esz ) { for( k = 0; k < esz; k++ ) dptr[k] = sptr[k]; } } } template static void randUniInt_(RNG& rng, _Tp* data, size_t total, int cn, const Scalar& scale, const Scalar& delta) { for( size_t i = 0; i < total; i += cn ) for( int k = 0; k < cn; k++ ) { int val = cvFloor( randInt(rng)*scale[k] + delta[k] ); data[i + k] = saturate_cast<_Tp>(val); } } template static void randUniFlt_(RNG& rng, _Tp* data, size_t total, int cn, const Scalar& scale, const Scalar& delta) { for( size_t i = 0; i < total; i += cn ) for( int k = 0; k < cn; k++ ) { double val = randReal(rng)*scale[k] + delta[k]; data[i + k] = saturate_cast<_Tp>(val); } } void randUni( RNG& rng, Mat& a, const Scalar& param0, const Scalar& param1 ) { Scalar scale = param0; Scalar delta = param1; double C = a.depth() < CV_32F ? 1./(65536.*65536.) : 1.; for( int k = 0; k < 4; k++ ) { double s = scale.val[k] - delta.val[k]; if( s >= 0 ) scale.val[k] = s; else { delta.val[k] = scale.val[k]; scale.val[k] = -s; } scale.val[k] *= C; } const Mat *arrays[]={&a, 0}; Mat plane; NAryMatIterator it(arrays, &plane); size_t i, nplanes = it.nplanes; int depth = a.depth(), cn = a.channels(); size_t total = plane.total()*cn; for( i = 0; i < nplanes; i++, ++it ) { switch( depth ) { case CV_8U: randUniInt_(rng, plane.ptr(), total, cn, scale, delta); break; case CV_8S: randUniInt_(rng, plane.ptr(), total, cn, scale, delta); break; case CV_16U: randUniInt_(rng, plane.ptr(), total, cn, scale, delta); break; case CV_16S: randUniInt_(rng, plane.ptr(), total, cn, scale, delta); break; case CV_32S: randUniInt_(rng, plane.ptr(), total, cn, scale, delta); break; case CV_32F: randUniFlt_(rng, plane.ptr(), total, cn, scale, delta); break; case CV_64F: randUniFlt_(rng, plane.ptr(), total, cn, scale, delta); break; default: CV_Assert(0); } } } template static void erode_(const Mat& src, Mat& dst, const vector& ofsvec) { int width = dst.cols*src.channels(), n = (int)ofsvec.size(); const int* ofs = &ofsvec[0]; for( int y = 0; y < dst.rows; y++ ) { const _Tp* sptr = src.ptr<_Tp>(y); _Tp* dptr = dst.ptr<_Tp>(y); for( int x = 0; x < width; x++ ) { _Tp result = sptr[x + ofs[0]]; for( int i = 1; i < n; i++ ) result = std::min(result, sptr[x + ofs[i]]); dptr[x] = result; } } } template static void dilate_(const Mat& src, Mat& dst, const vector& ofsvec) { int width = dst.cols*src.channels(), n = (int)ofsvec.size(); const int* ofs = &ofsvec[0]; for( int y = 0; y < dst.rows; y++ ) { const _Tp* sptr = src.ptr<_Tp>(y); _Tp* dptr = dst.ptr<_Tp>(y); for( int x = 0; x < width; x++ ) { _Tp result = sptr[x + ofs[0]]; for( int i = 1; i < n; i++ ) result = std::max(result, sptr[x + ofs[i]]); dptr[x] = result; } } } void erode(const Mat& _src, Mat& dst, const Mat& _kernel, Point anchor, int borderType, const Scalar& _borderValue) { //if( _src.type() == CV_16UC3 && _src.size() == Size(1, 2) ) // putchar('*'); Mat kernel = _kernel, src; Scalar borderValue = _borderValue; if( kernel.empty() ) kernel = Mat::ones(3, 3, CV_8U); else { CV_Assert( kernel.type() == CV_8U ); } if( anchor == Point(-1,-1) ) anchor = Point(kernel.cols/2, kernel.rows/2); if( borderType == BORDER_CONSTANT ) borderValue = getMaxVal(src.depth()); copyMakeBorder(_src, src, anchor.y, kernel.rows - anchor.y - 1, anchor.x, kernel.cols - anchor.x - 1, borderType, borderValue); dst.create( _src.size(), src.type() ); vector ofs; int step = (int)(src.step/src.elemSize1()), cn = src.channels(); for( int i = 0; i < kernel.rows; i++ ) for( int j = 0; j < kernel.cols; j++ ) if( kernel.at(i, j) != 0 ) ofs.push_back(i*step + j*cn); if( ofs.empty() ) ofs.push_back(anchor.y*step + anchor.x*cn); switch( src.depth() ) { case CV_8U: erode_(src, dst, ofs); break; case CV_8S: erode_(src, dst, ofs); break; case CV_16U: erode_(src, dst, ofs); break; case CV_16S: erode_(src, dst, ofs); break; case CV_32S: erode_(src, dst, ofs); break; case CV_32F: erode_(src, dst, ofs); break; case CV_64F: erode_(src, dst, ofs); break; default: CV_Assert(0); } } void dilate(const Mat& _src, Mat& dst, const Mat& _kernel, Point anchor, int borderType, const Scalar& _borderValue) { Mat kernel = _kernel, src; Scalar borderValue = _borderValue; if( kernel.empty() ) kernel = Mat::ones(3, 3, CV_8U); else { CV_Assert( kernel.type() == CV_8U ); } if( anchor == Point(-1,-1) ) anchor = Point(kernel.cols/2, kernel.rows/2); if( borderType == BORDER_CONSTANT ) borderValue = getMinVal(src.depth()); copyMakeBorder(_src, src, anchor.y, kernel.rows - anchor.y - 1, anchor.x, kernel.cols - anchor.x - 1, borderType, borderValue); dst.create( _src.size(), src.type() ); vector ofs; int step = (int)(src.step/src.elemSize1()), cn = src.channels(); for( int i = 0; i < kernel.rows; i++ ) for( int j = 0; j < kernel.cols; j++ ) if( kernel.at(i, j) != 0 ) ofs.push_back(i*step + j*cn); if( ofs.empty() ) ofs.push_back(anchor.y*step + anchor.x*cn); switch( src.depth() ) { case CV_8U: dilate_(src, dst, ofs); break; case CV_8S: dilate_(src, dst, ofs); break; case CV_16U: dilate_(src, dst, ofs); break; case CV_16S: dilate_(src, dst, ofs); break; case CV_32S: dilate_(src, dst, ofs); break; case CV_32F: dilate_(src, dst, ofs); break; case CV_64F: dilate_(src, dst, ofs); break; default: CV_Assert(0); } } template static void filter2D_(const Mat& src, Mat& dst, const vector& ofsvec, const vector& coeffvec) { const int* ofs = &ofsvec[0]; const double* coeff = &coeffvec[0]; int width = dst.cols*dst.channels(), ncoeffs = (int)ofsvec.size(); for( int y = 0; y < dst.rows; y++ ) { const _Tp* sptr = src.ptr<_Tp>(y); double* dptr = dst.ptr(y); for( int x = 0; x < width; x++ ) { double s = 0; for( int i = 0; i < ncoeffs; i++ ) s += sptr[x + ofs[i]]*coeff[i]; dptr[x] = s; } } } void filter2D(const Mat& _src, Mat& dst, int ddepth, const Mat& kernel, Point anchor, double delta, int borderType, const Scalar& _borderValue) { Mat src, _dst; Scalar borderValue = _borderValue; CV_Assert( kernel.type() == CV_32F || kernel.type() == CV_64F ); if( anchor == Point(-1,-1) ) anchor = Point(kernel.cols/2, kernel.rows/2); if( borderType == BORDER_CONSTANT ) borderValue = getMinVal(src.depth()); copyMakeBorder(_src, src, anchor.y, kernel.rows - anchor.y - 1, anchor.x, kernel.cols - anchor.x - 1, borderType, borderValue); _dst.create( _src.size(), CV_MAKETYPE(CV_64F, src.channels()) ); vector ofs; vector coeff(kernel.rows*kernel.cols); Mat cmat(kernel.rows, kernel.cols, CV_64F, &coeff[0]); convert(kernel, cmat, cmat.type()); int step = (int)(src.step/src.elemSize1()), cn = src.channels(); for( int i = 0; i < kernel.rows; i++ ) for( int j = 0; j < kernel.cols; j++ ) ofs.push_back(i*step + j*cn); switch( src.depth() ) { case CV_8U: filter2D_(src, _dst, ofs, coeff); break; case CV_8S: filter2D_(src, _dst, ofs, coeff); break; case CV_16U: filter2D_(src, _dst, ofs, coeff); break; case CV_16S: filter2D_(src, _dst, ofs, coeff); break; case CV_32S: filter2D_(src, _dst, ofs, coeff); break; case CV_32F: filter2D_(src, _dst, ofs, coeff); break; case CV_64F: filter2D_(src, _dst, ofs, coeff); break; default: CV_Assert(0); } convert(_dst, dst, ddepth, 1, delta); } static int borderInterpolate( int p, int len, int borderType ) { if( (unsigned)p < (unsigned)len ) ; else if( borderType == BORDER_REPLICATE ) p = p < 0 ? 0 : len - 1; else if( borderType == BORDER_REFLECT || borderType == BORDER_REFLECT_101 ) { int delta = borderType == BORDER_REFLECT_101; if( len == 1 ) return 0; do { if( p < 0 ) p = -p - 1 + delta; else p = len - 1 - (p - len) - delta; } while( (unsigned)p >= (unsigned)len ); } else if( borderType == BORDER_WRAP ) { if( p < 0 ) p -= ((p-len+1)/len)*len; if( p >= len ) p %= len; } else if( borderType == BORDER_CONSTANT ) p = -1; else CV_Error( Error::StsBadArg, "Unknown/unsupported border type" ); return p; } void copyMakeBorder(const Mat& src, Mat& dst, int top, int bottom, int left, int right, int borderType, const Scalar& borderValue) { dst.create(src.rows + top + bottom, src.cols + left + right, src.type()); int i, j, k, esz = (int)src.elemSize(); int width = src.cols*esz, width1 = dst.cols*esz; if( borderType == BORDER_CONSTANT ) { vector valvec((src.cols + left + right)*esz); uchar* val = &valvec[0]; scalarToRawData(borderValue, val, src.type(), (src.cols + left + right)*src.channels()); left *= esz; right *= esz; for( i = 0; i < src.rows; i++ ) { const uchar* sptr = src.ptr(i); uchar* dptr = dst.ptr(i + top) + left; for( j = 0; j < left; j++ ) dptr[j - left] = val[j]; if( dptr != sptr ) for( j = 0; j < width; j++ ) dptr[j] = sptr[j]; for( j = 0; j < right; j++ ) dptr[j + width] = val[j]; } for( i = 0; i < top; i++ ) { uchar* dptr = dst.ptr(i); for( j = 0; j < width1; j++ ) dptr[j] = val[j]; } for( i = 0; i < bottom; i++ ) { uchar* dptr = dst.ptr(i + top + src.rows); for( j = 0; j < width1; j++ ) dptr[j] = val[j]; } } else { vector tabvec((left + right)*esz + 1); int* ltab = &tabvec[0]; int* rtab = &tabvec[left*esz]; for( i = 0; i < left; i++ ) { j = borderInterpolate(i - left, src.cols, borderType)*esz; for( k = 0; k < esz; k++ ) ltab[i*esz + k] = j + k; } for( i = 0; i < right; i++ ) { j = borderInterpolate(src.cols + i, src.cols, borderType)*esz; for( k = 0; k < esz; k++ ) rtab[i*esz + k] = j + k; } left *= esz; right *= esz; for( i = 0; i < src.rows; i++ ) { const uchar* sptr = src.ptr(i); uchar* dptr = dst.ptr(i + top); for( j = 0; j < left; j++ ) dptr[j] = sptr[ltab[j]]; if( dptr + left != sptr ) { for( j = 0; j < width; j++ ) dptr[j + left] = sptr[j]; } for( j = 0; j < right; j++ ) dptr[j + left + width] = sptr[rtab[j]]; } for( i = 0; i < top; i++ ) { j = borderInterpolate(i - top, src.rows, borderType); const uchar* sptr = dst.ptr(j + top); uchar* dptr = dst.ptr(i); for( k = 0; k < width1; k++ ) dptr[k] = sptr[k]; } for( i = 0; i < bottom; i++ ) { j = borderInterpolate(i + src.rows, src.rows, borderType); const uchar* sptr = dst.ptr(j + top); uchar* dptr = dst.ptr(i + top + src.rows); for( k = 0; k < width1; k++ ) dptr[k] = sptr[k]; } } } template static void minMaxLoc_(const _Tp* src, size_t total, size_t startidx, double* _minval, double* _maxval, size_t* _minpos, size_t* _maxpos, const uchar* mask) { _Tp maxval = saturate_cast<_Tp>(*_maxval), minval = saturate_cast<_Tp>(*_minval); size_t minpos = *_minpos, maxpos = *_maxpos; if( !mask ) { for( size_t i = 0; i < total; i++ ) { _Tp val = src[i]; if( minval > val || !minpos ) { minval = val; minpos = startidx + i; } if( maxval < val || !maxpos ) { maxval = val; maxpos = startidx + i; } } } else { for( size_t i = 0; i < total; i++ ) { _Tp val = src[i]; if( (minval > val || !minpos) && mask[i] ) { minval = val; minpos = startidx + i; } if( (maxval < val || !maxpos) && mask[i] ) { maxval = val; maxpos = startidx + i; } } } *_maxval = maxval; *_minval = minval; *_maxpos = maxpos; *_minpos = minpos; } static void setpos( const Mat& mtx, vector& pos, size_t idx ) { pos.resize(mtx.dims); if( idx > 0 ) { idx--; for( int i = mtx.dims-1; i >= 0; i-- ) { int sz = mtx.size[i]*(i == mtx.dims-1 ? mtx.channels() : 1); pos[i] = (int)(idx % sz); idx /= sz; } } else { for( int i = mtx.dims-1; i >= 0; i-- ) pos[i] = -1; } } void minMaxLoc(const Mat& src, double* _minval, double* _maxval, vector* _minloc, vector* _maxloc, const Mat& mask) { CV_Assert( src.channels() == 1 ); const Mat *arrays[]={&src, &mask, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t startidx = 1, total = planes[0].total(); size_t i, nplanes = it.nplanes; int depth = src.depth(); double minval = 0; double maxval = 0; size_t maxidx = 0, minidx = 0; for( i = 0; i < nplanes; i++, ++it, startidx += total ) { const uchar* sptr = planes[0].ptr(); const uchar* mptr = planes[1].ptr(); switch( depth ) { case CV_8U: minMaxLoc_((const uchar*)sptr, total, startidx, &minval, &maxval, &minidx, &maxidx, mptr); break; case CV_8S: minMaxLoc_((const schar*)sptr, total, startidx, &minval, &maxval, &minidx, &maxidx, mptr); break; case CV_16U: minMaxLoc_((const ushort*)sptr, total, startidx, &minval, &maxval, &minidx, &maxidx, mptr); break; case CV_16S: minMaxLoc_((const short*)sptr, total, startidx, &minval, &maxval, &minidx, &maxidx, mptr); break; case CV_32S: minMaxLoc_((const int*)sptr, total, startidx, &minval, &maxval, &minidx, &maxidx, mptr); break; case CV_32F: minMaxLoc_((const float*)sptr, total, startidx, &minval, &maxval, &minidx, &maxidx, mptr); break; case CV_64F: minMaxLoc_((const double*)sptr, total, startidx, &minval, &maxval, &minidx, &maxidx, mptr); break; default: CV_Assert(0); } } if( _maxval ) *_maxval = maxval; if( _minval ) *_minval = minval; if( _maxloc ) setpos( src, *_maxloc, maxidx ); if( _minloc ) setpos( src, *_minloc, minidx ); } static int normHamming(const uchar* src, size_t total, int cellSize) { int result = 0; int mask = cellSize == 1 ? 1 : cellSize == 2 ? 3 : cellSize == 4 ? 15 : -1; CV_Assert( mask >= 0 ); for( size_t i = 0; i < total; i++ ) { unsigned a = src[i]; for( ; a != 0; a >>= cellSize ) result += (a & mask) != 0; } return result; } template static double norm_(const _Tp* src, size_t total, int cn, int normType, double startval, const uchar* mask) { size_t i; double result = startval; if( !mask ) total *= cn; if( normType == NORM_INF ) { if( !mask ) for( i = 0; i < total; i++ ) result = std::max(result, (double)std::abs(0+src[i]));// trick with 0 used to quiet gcc warning else for( int c = 0; c < cn; c++ ) { for( i = 0; i < total; i++ ) if( mask[i] ) result = std::max(result, (double)std::abs(0+src[i*cn + c])); } } else if( normType == NORM_L1 ) { if( !mask ) for( i = 0; i < total; i++ ) result += std::abs(0+src[i]); else for( int c = 0; c < cn; c++ ) { for( i = 0; i < total; i++ ) if( mask[i] ) result += std::abs(0+src[i*cn + c]); } } else { if( !mask ) for( i = 0; i < total; i++ ) { double v = src[i]; result += v*v; } else for( int c = 0; c < cn; c++ ) { for( i = 0; i < total; i++ ) if( mask[i] ) { double v = src[i*cn + c]; result += v*v; } } } return result; } template static double norm_(const _Tp* src1, const _Tp* src2, size_t total, int cn, int normType, double startval, const uchar* mask) { size_t i; double result = startval; if( !mask ) total *= cn; if( normType == NORM_INF ) { if( !mask ) for( i = 0; i < total; i++ ) result = std::max(result, (double)std::abs(src1[i] - src2[i])); else for( int c = 0; c < cn; c++ ) { for( i = 0; i < total; i++ ) if( mask[i] ) result = std::max(result, (double)std::abs(src1[i*cn + c] - src2[i*cn + c])); } } else if( normType == NORM_L1 ) { if( !mask ) for( i = 0; i < total; i++ ) result += std::abs(src1[i] - src2[i]); else for( int c = 0; c < cn; c++ ) { for( i = 0; i < total; i++ ) if( mask[i] ) result += std::abs(src1[i*cn + c] - src2[i*cn + c]); } } else { if( !mask ) for( i = 0; i < total; i++ ) { double v = src1[i] - src2[i]; result += v*v; } else for( int c = 0; c < cn; c++ ) { for( i = 0; i < total; i++ ) if( mask[i] ) { double v = src1[i*cn + c] - src2[i*cn + c]; result += v*v; } } } return result; } double norm(InputArray _src, int normType, InputArray _mask) { Mat src = _src.getMat(), mask = _mask.getMat(); if( src.depth() == CV_16F ) { Mat src32f; src.convertTo(src32f, CV_32F); return cvtest::norm(src32f, normType, _mask); } if( normType == NORM_HAMMING || normType == NORM_HAMMING2 ) { if( !mask.empty() ) { Mat temp; bitwise_and(src, mask, temp); return cvtest::norm(temp, normType, Mat()); } CV_Assert( src.depth() == CV_8U ); const Mat *arrays[]={&src, 0}; Mat planes[1]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; double result = 0; int cellSize = normType == NORM_HAMMING ? 1 : 2; for( i = 0; i < nplanes; i++, ++it ) result += normHamming(planes[0].ptr(), total, cellSize); return result; } int normType0 = normType; normType = normType == NORM_L2SQR ? NORM_L2 : normType; CV_Assert( mask.empty() || (src.size == mask.size && mask.type() == CV_8U) ); CV_Assert( normType == NORM_INF || normType == NORM_L1 || normType == NORM_L2 ); const Mat *arrays[]={&src, &mask, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; int depth = src.depth(), cn = planes[0].channels(); double result = 0; for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr = planes[0].ptr(); const uchar* mptr = planes[1].ptr(); switch( depth ) { case CV_8U: result = norm_((const uchar*)sptr, total, cn, normType, result, mptr); break; case CV_8S: result = norm_((const schar*)sptr, total, cn, normType, result, mptr); break; case CV_16U: result = norm_((const ushort*)sptr, total, cn, normType, result, mptr); break; case CV_16S: result = norm_((const short*)sptr, total, cn, normType, result, mptr); break; case CV_32S: result = norm_((const int*)sptr, total, cn, normType, result, mptr); break; case CV_32F: result = norm_((const float*)sptr, total, cn, normType, result, mptr); break; case CV_64F: result = norm_((const double*)sptr, total, cn, normType, result, mptr); break; default: CV_Error(Error::StsUnsupportedFormat, ""); }; } if( normType0 == NORM_L2 ) result = sqrt(result); return result; } double norm(InputArray _src1, InputArray _src2, int normType, InputArray _mask) { Mat src1 = _src1.getMat(), src2 = _src2.getMat(), mask = _mask.getMat(); if( src1.depth() == CV_16F ) { Mat src1_32f, src2_32f; src1.convertTo(src1_32f, CV_32F); src2.convertTo(src2_32f, CV_32F); return cvtest::norm(src1_32f, src2_32f, normType, _mask); } bool isRelative = (normType & NORM_RELATIVE) != 0; normType &= ~NORM_RELATIVE; if( normType == NORM_HAMMING || normType == NORM_HAMMING2 ) { Mat temp; bitwise_xor(src1, src2, temp); if( !mask.empty() ) bitwise_and(temp, mask, temp); CV_Assert( temp.depth() == CV_8U ); const Mat *arrays[]={&temp, 0}; Mat planes[1]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; double result = 0; int cellSize = normType == NORM_HAMMING ? 1 : 2; for( i = 0; i < nplanes; i++, ++it ) result += normHamming(planes[0].ptr(), total, cellSize); return result; } int normType0 = normType; normType = normType == NORM_L2SQR ? NORM_L2 : normType; CV_Assert( src1.type() == src2.type() && src1.size == src2.size ); CV_Assert( mask.empty() || (src1.size == mask.size && mask.type() == CV_8U) ); CV_Assert( normType == NORM_INF || normType == NORM_L1 || normType == NORM_L2 ); const Mat *arrays[]={&src1, &src2, &mask, 0}; Mat planes[3]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; int depth = src1.depth(), cn = planes[0].channels(); double result = 0; for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); const uchar* mptr = planes[2].ptr(); switch( depth ) { case CV_8U: result = norm_((const uchar*)sptr1, (const uchar*)sptr2, total, cn, normType, result, mptr); break; case CV_8S: result = norm_((const schar*)sptr1, (const schar*)sptr2, total, cn, normType, result, mptr); break; case CV_16U: result = norm_((const ushort*)sptr1, (const ushort*)sptr2, total, cn, normType, result, mptr); break; case CV_16S: result = norm_((const short*)sptr1, (const short*)sptr2, total, cn, normType, result, mptr); break; case CV_32S: result = norm_((const int*)sptr1, (const int*)sptr2, total, cn, normType, result, mptr); break; case CV_32F: result = norm_((const float*)sptr1, (const float*)sptr2, total, cn, normType, result, mptr); break; case CV_64F: result = norm_((const double*)sptr1, (const double*)sptr2, total, cn, normType, result, mptr); break; default: CV_Error(Error::StsUnsupportedFormat, ""); }; } if( normType0 == NORM_L2 ) result = sqrt(result); return isRelative ? result / (cvtest::norm(src2, normType) + DBL_EPSILON) : result; } double PSNR(InputArray _src1, InputArray _src2) { CV_Assert( _src1.depth() == CV_8U ); double diff = std::sqrt(cvtest::norm(_src1, _src2, NORM_L2SQR)/(_src1.total()*_src1.channels())); return 20*log10(255./(diff+DBL_EPSILON)); } template static double crossCorr_(const _Tp* src1, const _Tp* src2, size_t total) { double result = 0; for( size_t i = 0; i < total; i++ ) result += (double)src1[i]*src2[i]; return result; } double crossCorr(const Mat& src1, const Mat& src2) { CV_Assert( src1.size == src2.size && src1.type() == src2.type() ); const Mat *arrays[]={&src1, &src2, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].channels(); size_t i, nplanes = it.nplanes; int depth = src1.depth(); double result = 0; for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); switch( depth ) { case CV_8U: result += crossCorr_((const uchar*)sptr1, (const uchar*)sptr2, total); break; case CV_8S: result += crossCorr_((const schar*)sptr1, (const schar*)sptr2, total); break; case CV_16U: result += crossCorr_((const ushort*)sptr1, (const ushort*)sptr2, total); break; case CV_16S: result += crossCorr_((const short*)sptr1, (const short*)sptr2, total); break; case CV_32S: result += crossCorr_((const int*)sptr1, (const int*)sptr2, total); break; case CV_32F: result += crossCorr_((const float*)sptr1, (const float*)sptr2, total); break; case CV_64F: result += crossCorr_((const double*)sptr1, (const double*)sptr2, total); break; default: CV_Error(Error::StsUnsupportedFormat, ""); }; } return result; } static void logicOp_(const uchar* src1, const uchar* src2, uchar* dst, size_t total, char c) { size_t i; if( c == '&' ) for( i = 0; i < total; i++ ) dst[i] = src1[i] & src2[i]; else if( c == '|' ) for( i = 0; i < total; i++ ) dst[i] = src1[i] | src2[i]; else for( i = 0; i < total; i++ ) dst[i] = src1[i] ^ src2[i]; } static void logicOpS_(const uchar* src, const uchar* scalar, uchar* dst, size_t total, char c) { const size_t blockSize = 96; size_t i, j; if( c == '&' ) for( i = 0; i < total; i += blockSize, dst += blockSize, src += blockSize ) { size_t sz = MIN(total - i, blockSize); for( j = 0; j < sz; j++ ) dst[j] = src[j] & scalar[j]; } else if( c == '|' ) for( i = 0; i < total; i += blockSize, dst += blockSize, src += blockSize ) { size_t sz = MIN(total - i, blockSize); for( j = 0; j < sz; j++ ) dst[j] = src[j] | scalar[j]; } else if( c == '^' ) { for( i = 0; i < total; i += blockSize, dst += blockSize, src += blockSize ) { size_t sz = MIN(total - i, blockSize); for( j = 0; j < sz; j++ ) dst[j] = src[j] ^ scalar[j]; } } else for( i = 0; i < total; i++ ) dst[i] = ~src[i]; } void logicOp( const Mat& src1, const Mat& src2, Mat& dst, char op ) { CV_Assert( op == '&' || op == '|' || op == '^' ); CV_Assert( src1.type() == src2.type() && src1.size == src2.size ); dst.create( src1.dims, &src1.size[0], src1.type() ); const Mat *arrays[]={&src1, &src2, &dst, 0}; Mat planes[3]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].elemSize(); size_t i, nplanes = it.nplanes; for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); uchar* dptr = planes[2].ptr(); logicOp_(sptr1, sptr2, dptr, total, op); } } void logicOp(const Mat& src, const Scalar& s, Mat& dst, char op) { CV_Assert( op == '&' || op == '|' || op == '^' || op == '~' ); dst.create( src.dims, &src.size[0], src.type() ); const Mat *arrays[]={&src, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].elemSize(); size_t i, nplanes = it.nplanes; double buf[12]; scalarToRawData(s, buf, src.type(), (int)(96/planes[0].elemSize1())); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr = planes[0].ptr(); uchar* dptr = planes[1].ptr(); logicOpS_(sptr, (uchar*)&buf[0], dptr, total, op); } } template static void compare_(const _Tp* src1, const _Tp* src2, uchar* dst, size_t total, int cmpop) { size_t i; switch( cmpop ) { case CMP_LT: for( i = 0; i < total; i++ ) dst[i] = src1[i] < src2[i] ? 255 : 0; break; case CMP_LE: for( i = 0; i < total; i++ ) dst[i] = src1[i] <= src2[i] ? 255 : 0; break; case CMP_EQ: for( i = 0; i < total; i++ ) dst[i] = src1[i] == src2[i] ? 255 : 0; break; case CMP_NE: for( i = 0; i < total; i++ ) dst[i] = src1[i] != src2[i] ? 255 : 0; break; case CMP_GE: for( i = 0; i < total; i++ ) dst[i] = src1[i] >= src2[i] ? 255 : 0; break; case CMP_GT: for( i = 0; i < total; i++ ) dst[i] = src1[i] > src2[i] ? 255 : 0; break; default: CV_Error(Error::StsBadArg, "Unknown comparison operation"); } } template static void compareS_(const _Tp* src1, _WTp value, uchar* dst, size_t total, int cmpop) { size_t i; switch( cmpop ) { case CMP_LT: for( i = 0; i < total; i++ ) dst[i] = src1[i] < value ? 255 : 0; break; case CMP_LE: for( i = 0; i < total; i++ ) dst[i] = src1[i] <= value ? 255 : 0; break; case CMP_EQ: for( i = 0; i < total; i++ ) dst[i] = src1[i] == value ? 255 : 0; break; case CMP_NE: for( i = 0; i < total; i++ ) dst[i] = src1[i] != value ? 255 : 0; break; case CMP_GE: for( i = 0; i < total; i++ ) dst[i] = src1[i] >= value ? 255 : 0; break; case CMP_GT: for( i = 0; i < total; i++ ) dst[i] = src1[i] > value ? 255 : 0; break; default: CV_Error(Error::StsBadArg, "Unknown comparison operation"); } } void compare(const Mat& src1, const Mat& src2, Mat& dst, int cmpop) { CV_Assert( src1.type() == src2.type() && src1.channels() == 1 && src1.size == src2.size ); dst.create( src1.dims, &src1.size[0], CV_8U ); const Mat *arrays[]={&src1, &src2, &dst, 0}; Mat planes[3]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; int depth = src1.depth(); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); uchar* dptr = planes[2].ptr(); switch( depth ) { case CV_8U: compare_((const uchar*)sptr1, (const uchar*)sptr2, dptr, total, cmpop); break; case CV_8S: compare_((const schar*)sptr1, (const schar*)sptr2, dptr, total, cmpop); break; case CV_16U: compare_((const ushort*)sptr1, (const ushort*)sptr2, dptr, total, cmpop); break; case CV_16S: compare_((const short*)sptr1, (const short*)sptr2, dptr, total, cmpop); break; case CV_32S: compare_((const int*)sptr1, (const int*)sptr2, dptr, total, cmpop); break; case CV_32F: compare_((const float*)sptr1, (const float*)sptr2, dptr, total, cmpop); break; case CV_64F: compare_((const double*)sptr1, (const double*)sptr2, dptr, total, cmpop); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } } } void compare(const Mat& src, double value, Mat& dst, int cmpop) { CV_Assert( src.channels() == 1 ); dst.create( src.dims, &src.size[0], CV_8U ); const Mat *arrays[]={&src, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; int depth = src.depth(); int ivalue = saturate_cast(value); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr = planes[0].ptr(); uchar* dptr = planes[1].ptr(); switch( depth ) { case CV_8U: compareS_((const uchar*)sptr, ivalue, dptr, total, cmpop); break; case CV_8S: compareS_((const schar*)sptr, ivalue, dptr, total, cmpop); break; case CV_16U: compareS_((const ushort*)sptr, ivalue, dptr, total, cmpop); break; case CV_16S: compareS_((const short*)sptr, ivalue, dptr, total, cmpop); break; case CV_32S: compareS_((const int*)sptr, ivalue, dptr, total, cmpop); break; case CV_32F: compareS_((const float*)sptr, value, dptr, total, cmpop); break; case CV_64F: compareS_((const double*)sptr, value, dptr, total, cmpop); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } } } template double cmpUlpsInt_(const _Tp* src1, const _Tp* src2, size_t total, int imaxdiff, size_t startidx, size_t& idx) { size_t i; int realmaxdiff = 0; for( i = 0; i < total; i++ ) { int diff = std::abs(src1[i] - src2[i]); if( realmaxdiff < diff ) { realmaxdiff = diff; if( diff > imaxdiff && idx == 0 ) idx = i + startidx; } } return realmaxdiff; } template<> double cmpUlpsInt_(const int* src1, const int* src2, size_t total, int imaxdiff, size_t startidx, size_t& idx) { size_t i; double realmaxdiff = 0; for( i = 0; i < total; i++ ) { double diff = fabs((double)src1[i] - (double)src2[i]); if( realmaxdiff < diff ) { realmaxdiff = diff; if( diff > imaxdiff && idx == 0 ) idx = i + startidx; } } return realmaxdiff; } static double cmpUlpsFlt_(const int* src1, const int* src2, size_t total, int imaxdiff, size_t startidx, size_t& idx) { const int C = 0x7fffffff; int realmaxdiff = 0; size_t i; for( i = 0; i < total; i++ ) { int a = src1[i], b = src2[i]; if( a < 0 ) a ^= C; if( b < 0 ) b ^= C; int diff = std::abs(a - b); if( realmaxdiff < diff ) { realmaxdiff = diff; if( diff > imaxdiff && idx == 0 ) idx = i + startidx; } } return realmaxdiff; } static double cmpUlpsFlt_(const int64* src1, const int64* src2, size_t total, int imaxdiff, size_t startidx, size_t& idx) { const int64 C = CV_BIG_INT(0x7fffffffffffffff); double realmaxdiff = 0; size_t i; for( i = 0; i < total; i++ ) { int64 a = src1[i], b = src2[i]; if( a < 0 ) a ^= C; if( b < 0 ) b ^= C; double diff = fabs((double)a - (double)b); if( realmaxdiff < diff ) { realmaxdiff = diff; if( diff > imaxdiff && idx == 0 ) idx = i + startidx; } } return realmaxdiff; } bool cmpUlps(const Mat& src1, const Mat& src2, int imaxDiff, double* _realmaxdiff, vector* loc) { CV_Assert( src1.type() == src2.type() && src1.size == src2.size ); const Mat *arrays[]={&src1, &src2, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].channels(); size_t i, nplanes = it.nplanes; int depth = src1.depth(); size_t startidx = 1, idx = 0; if(_realmaxdiff) *_realmaxdiff = 0; for( i = 0; i < nplanes; i++, ++it, startidx += total ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); double realmaxdiff = 0; switch( depth ) { case CV_8U: realmaxdiff = cmpUlpsInt_((const uchar*)sptr1, (const uchar*)sptr2, total, imaxDiff, startidx, idx); break; case CV_8S: realmaxdiff = cmpUlpsInt_((const schar*)sptr1, (const schar*)sptr2, total, imaxDiff, startidx, idx); break; case CV_16U: realmaxdiff = cmpUlpsInt_((const ushort*)sptr1, (const ushort*)sptr2, total, imaxDiff, startidx, idx); break; case CV_16S: realmaxdiff = cmpUlpsInt_((const short*)sptr1, (const short*)sptr2, total, imaxDiff, startidx, idx); break; case CV_32S: realmaxdiff = cmpUlpsInt_((const int*)sptr1, (const int*)sptr2, total, imaxDiff, startidx, idx); break; case CV_32F: realmaxdiff = cmpUlpsFlt_((const int*)sptr1, (const int*)sptr2, total, imaxDiff, startidx, idx); break; case CV_64F: realmaxdiff = cmpUlpsFlt_((const int64*)sptr1, (const int64*)sptr2, total, imaxDiff, startidx, idx); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } if(_realmaxdiff) *_realmaxdiff = std::max(*_realmaxdiff, realmaxdiff); } if(idx > 0 && loc) setpos(src1, *loc, idx); return idx == 0; } template static void checkInt_(const _Tp* a, size_t total, int imin, int imax, size_t startidx, size_t& idx) { for( size_t i = 0; i < total; i++ ) { int val = a[i]; if( val < imin || val > imax ) { idx = i + startidx; break; } } } template static void checkFlt_(const _Tp* a, size_t total, double fmin, double fmax, size_t startidx, size_t& idx) { for( size_t i = 0; i < total; i++ ) { double val = a[i]; if( cvIsNaN(val) || cvIsInf(val) || val < fmin || val > fmax ) { idx = i + startidx; break; } } } // checks that the array does not have NaNs and/or Infs and all the elements are // within [min_val,max_val). idx is the index of the first "bad" element. int check( const Mat& a, double fmin, double fmax, vector* _idx ) { const Mat *arrays[]={&a, 0}; Mat plane; NAryMatIterator it(arrays, &plane); size_t total = plane.total()*plane.channels(); size_t i, nplanes = it.nplanes; int depth = a.depth(); size_t startidx = 1, idx = 0; int imin = 0, imax = 0; if( depth <= CV_32S ) { imin = cvCeil(fmin); imax = cvFloor(fmax); } for( i = 0; i < nplanes; i++, ++it, startidx += total ) { const uchar* aptr = plane.ptr(); switch( depth ) { case CV_8U: checkInt_((const uchar*)aptr, total, imin, imax, startidx, idx); break; case CV_8S: checkInt_((const schar*)aptr, total, imin, imax, startidx, idx); break; case CV_16U: checkInt_((const ushort*)aptr, total, imin, imax, startidx, idx); break; case CV_16S: checkInt_((const short*)aptr, total, imin, imax, startidx, idx); break; case CV_32S: checkInt_((const int*)aptr, total, imin, imax, startidx, idx); break; case CV_32F: checkFlt_((const float*)aptr, total, fmin, fmax, startidx, idx); break; case CV_64F: checkFlt_((const double*)aptr, total, fmin, fmax, startidx, idx); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } if( idx != 0 ) break; } if(idx != 0 && _idx) setpos(a, *_idx, idx); return idx == 0 ? 0 : -1; } #define CMP_EPS_OK 0 #define CMP_EPS_BIG_DIFF -1 #define CMP_EPS_INVALID_TEST_DATA -2 // there is NaN or Inf value in test data #define CMP_EPS_INVALID_REF_DATA -3 // there is NaN or Inf value in reference data // compares two arrays. max_diff is the maximum actual difference, // success_err_level is maximum allowed difference, idx is the index of the first // element for which difference is >success_err_level // (or index of element with the maximum difference) int cmpEps( const Mat& arr_, const Mat& refarr_, double* _realmaxdiff, double success_err_level, vector* _idx, bool element_wise_relative_error ) { Mat arr = arr_, refarr = refarr_; CV_Assert( arr.type() == refarr.type() && arr.size == refarr.size ); if( arr.depth() == CV_16F ) { Mat arr32f, refarr32f; arr.convertTo(arr32f, CV_32F); refarr.convertTo(refarr32f, CV_32F); arr = arr32f; refarr = refarr32f; } int ilevel = refarr.depth() <= CV_32S ? cvFloor(success_err_level) : 0; int result = CMP_EPS_OK; const Mat *arrays[]={&arr, &refarr, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].channels(), j = total; size_t i, nplanes = it.nplanes; int depth = arr.depth(); size_t startidx = 1, idx = 0; double realmaxdiff = 0, maxval = 0; if(_realmaxdiff) *_realmaxdiff = 0; if( refarr.depth() >= CV_32F && !element_wise_relative_error ) { maxval = cvtest::norm( refarr, NORM_INF ); maxval = MAX(maxval, 1.); } for( i = 0; i < nplanes; i++, ++it, startidx += total ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); switch( depth ) { case CV_8U: realmaxdiff = cmpUlpsInt_((const uchar*)sptr1, (const uchar*)sptr2, total, ilevel, startidx, idx); break; case CV_8S: realmaxdiff = cmpUlpsInt_((const schar*)sptr1, (const schar*)sptr2, total, ilevel, startidx, idx); break; case CV_16U: realmaxdiff = cmpUlpsInt_((const ushort*)sptr1, (const ushort*)sptr2, total, ilevel, startidx, idx); break; case CV_16S: realmaxdiff = cmpUlpsInt_((const short*)sptr1, (const short*)sptr2, total, ilevel, startidx, idx); break; case CV_32S: realmaxdiff = cmpUlpsInt_((const int*)sptr1, (const int*)sptr2, total, ilevel, startidx, idx); break; case CV_32F: for( j = 0; j < total; j++ ) { double a_val = ((float*)sptr1)[j]; double b_val = ((float*)sptr2)[j]; double threshold; if( ((int*)sptr1)[j] == ((int*)sptr2)[j] ) continue; if( cvIsNaN(a_val) || cvIsInf(a_val) ) { result = CMP_EPS_INVALID_TEST_DATA; idx = startidx + j; break; } if( cvIsNaN(b_val) || cvIsInf(b_val) ) { result = CMP_EPS_INVALID_REF_DATA; idx = startidx + j; break; } a_val = fabs(a_val - b_val); threshold = element_wise_relative_error ? fabs(b_val) + 1 : maxval; if( a_val > threshold*success_err_level ) { realmaxdiff = a_val/threshold; if( idx == 0 ) idx = startidx + j; break; } } break; case CV_64F: for( j = 0; j < total; j++ ) { double a_val = ((double*)sptr1)[j]; double b_val = ((double*)sptr2)[j]; double threshold; if( ((int64*)sptr1)[j] == ((int64*)sptr2)[j] ) continue; if( cvIsNaN(a_val) || cvIsInf(a_val) ) { result = CMP_EPS_INVALID_TEST_DATA; idx = startidx + j; break; } if( cvIsNaN(b_val) || cvIsInf(b_val) ) { result = CMP_EPS_INVALID_REF_DATA; idx = startidx + j; break; } a_val = fabs(a_val - b_val); threshold = element_wise_relative_error ? fabs(b_val) + 1 : maxval; if( a_val > threshold*success_err_level ) { realmaxdiff = a_val/threshold; idx = startidx + j; break; } } break; default: assert(0); return CMP_EPS_BIG_DIFF; } if(_realmaxdiff) *_realmaxdiff = MAX(*_realmaxdiff, realmaxdiff); if( idx != 0 ) break; } if( result == 0 && idx != 0 ) result = CMP_EPS_BIG_DIFF; if( result < -1 && _realmaxdiff ) *_realmaxdiff = exp(1000.); if(idx > 0 && _idx) setpos(arr, *_idx, idx); return result; } int cmpEps2( TS* ts, const Mat& a, const Mat& b, double success_err_level, bool element_wise_relative_error, const char* desc ) { char msg[100]; double diff = 0; vector idx; int code = cmpEps( a, b, &diff, success_err_level, &idx, element_wise_relative_error ); switch( code ) { case CMP_EPS_BIG_DIFF: sprintf( msg, "%s: Too big difference (=%g > %g)", desc, diff, success_err_level ); code = TS::FAIL_BAD_ACCURACY; break; case CMP_EPS_INVALID_TEST_DATA: sprintf( msg, "%s: Invalid output", desc ); code = TS::FAIL_INVALID_OUTPUT; break; case CMP_EPS_INVALID_REF_DATA: sprintf( msg, "%s: Invalid reference output", desc ); code = TS::FAIL_INVALID_OUTPUT; break; default: ; } if( code < 0 ) { if( a.total() == 1 ) { ts->printf( TS::LOG, "%s\n", msg ); } else if( a.dims == 2 && (a.rows == 1 || a.cols == 1) ) { ts->printf( TS::LOG, "%s at element %d\n", msg, idx[0] + idx[1] ); } else { string idxstr = vec2str(", ", &idx[0], idx.size()); ts->printf( TS::LOG, "%s at (%s)\n", msg, idxstr.c_str() ); } } return code; } int cmpEps2_64f( TS* ts, const double* val, const double* refval, int len, double eps, const char* param_name ) { Mat _val(1, len, CV_64F, (void*)val); Mat _refval(1, len, CV_64F, (void*)refval); return cmpEps2( ts, _val, _refval, eps, true, param_name ); } template static void GEMM_(const _Tp* a_data0, int a_step, int a_delta, const _Tp* b_data0, int b_step, int b_delta, const _Tp* c_data0, int c_step, int c_delta, _Tp* d_data, int d_step, int d_rows, int d_cols, int a_cols, int cn, double alpha, double beta) { for( int i = 0; i < d_rows; i++, d_data += d_step, c_data0 += c_step, a_data0 += a_step ) { for( int j = 0; j < d_cols; j++ ) { const _Tp* a_data = a_data0; const _Tp* b_data = b_data0 + j*b_delta; const _Tp* c_data = c_data0 + j*c_delta; if( cn == 1 ) { double s = 0; for( int k = 0; k < a_cols; k++ ) { s += ((double)a_data[0])*b_data[0]; a_data += a_delta; b_data += b_step; } d_data[j] = (_Tp)(s*alpha + (c_data ? c_data[0]*beta : 0)); } else { double s_re = 0, s_im = 0; for( int k = 0; k < a_cols; k++ ) { s_re += ((double)a_data[0])*b_data[0] - ((double)a_data[1])*b_data[1]; s_im += ((double)a_data[0])*b_data[1] + ((double)a_data[1])*b_data[0]; a_data += a_delta; b_data += b_step; } s_re *= alpha; s_im *= alpha; if( c_data ) { s_re += c_data[0]*beta; s_im += c_data[1]*beta; } d_data[j*2] = (_Tp)s_re; d_data[j*2+1] = (_Tp)s_im; } } } } void gemm( const Mat& _a, const Mat& _b, double alpha, const Mat& _c, double beta, Mat& d, int flags ) { Mat a = _a, b = _b, c = _c; if( a.data == d.data ) a = a.clone(); if( b.data == d.data ) b = b.clone(); if( !c.empty() && c.data == d.data && (flags & cv::GEMM_3_T) ) c = c.clone(); int a_rows = a.rows, a_cols = a.cols, b_rows = b.rows, b_cols = b.cols; int cn = a.channels(); int a_step = (int)a.step1(), a_delta = cn; int b_step = (int)b.step1(), b_delta = cn; int c_rows = 0, c_cols = 0, c_step = 0, c_delta = 0; CV_Assert( a.type() == b.type() && a.dims == 2 && b.dims == 2 && cn <= 2 ); if( flags & cv::GEMM_1_T ) { std::swap( a_rows, a_cols ); std::swap( a_step, a_delta ); } if( flags & cv::GEMM_2_T ) { std::swap( b_rows, b_cols ); std::swap( b_step, b_delta ); } if( !c.empty() ) { c_rows = c.rows; c_cols = c.cols; c_step = (int)c.step1(); c_delta = cn; if( flags & cv::GEMM_3_T ) { std::swap( c_rows, c_cols ); std::swap( c_step, c_delta ); } CV_Assert( c.dims == 2 && c.type() == a.type() && c_rows == a_rows && c_cols == b_cols ); } d.create(a_rows, b_cols, a.type()); if( a.depth() == CV_32F ) GEMM_(a.ptr(), a_step, a_delta, b.ptr(), b_step, b_delta, !c.empty() ? c.ptr() : 0, c_step, c_delta, d.ptr(), (int)d.step1(), a_rows, b_cols, a_cols, cn, alpha, beta ); else GEMM_(a.ptr(), a_step, a_delta, b.ptr(), b_step, b_delta, !c.empty() ? c.ptr() : 0, c_step, c_delta, d.ptr(), (int)d.step1(), a_rows, b_cols, a_cols, cn, alpha, beta ); } template static void transform_(const _Tp* sptr, _Tp* dptr, size_t total, int scn, int dcn, const double* mat) { for( size_t i = 0; i < total; i++, sptr += scn, dptr += dcn ) { for( int j = 0; j < dcn; j++ ) { double s = mat[j*(scn + 1) + scn]; for( int k = 0; k < scn; k++ ) s += mat[j*(scn + 1) + k]*sptr[k]; dptr[j] = saturate_cast<_Tp>(s); } } } void transform( const Mat& src, Mat& dst, const Mat& transmat, const Mat& _shift ) { double mat[20]; int scn = src.channels(); int dcn = dst.channels(); int depth = src.depth(); int mattype = transmat.depth(); Mat shift = _shift.reshape(1, 0); bool haveShift = !shift.empty(); CV_Assert( scn <= 4 && dcn <= 4 && (mattype == CV_32F || mattype == CV_64F) && (!haveShift || (shift.type() == mattype && (shift.rows == 1 || shift.cols == 1))) ); // prepare cn x (cn + 1) transform matrix if( mattype == CV_32F ) { for( int i = 0; i < transmat.rows; i++ ) { mat[i*(scn+1)+scn] = 0.; for( int j = 0; j < transmat.cols; j++ ) mat[i*(scn+1)+j] = transmat.at(i,j); if( haveShift ) mat[i*(scn+1)+scn] = shift.at(i); } } else { for( int i = 0; i < transmat.rows; i++ ) { mat[i*(scn+1)+scn] = 0.; for( int j = 0; j < transmat.cols; j++ ) mat[i*(scn+1)+j] = transmat.at(i,j); if( haveShift ) mat[i*(scn+1)+scn] = shift.at(i); } } const Mat *arrays[]={&src, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr = planes[0].ptr(); uchar* dptr = planes[1].ptr(); switch( depth ) { case CV_8U: transform_((const uchar*)sptr, (uchar*)dptr, total, scn, dcn, mat); break; case CV_8S: transform_((const schar*)sptr, (schar*)dptr, total, scn, dcn, mat); break; case CV_16U: transform_((const ushort*)sptr, (ushort*)dptr, total, scn, dcn, mat); break; case CV_16S: transform_((const short*)sptr, (short*)dptr, total, scn, dcn, mat); break; case CV_32S: transform_((const int*)sptr, (int*)dptr, total, scn, dcn, mat); break; case CV_32F: transform_((const float*)sptr, (float*)dptr, total, scn, dcn, mat); break; case CV_64F: transform_((const double*)sptr, (double*)dptr, total, scn, dcn, mat); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } } } template static void minmax_(const _Tp* src1, const _Tp* src2, _Tp* dst, size_t total, char op) { if( op == 'M' ) for( size_t i = 0; i < total; i++ ) dst[i] = std::max(src1[i], src2[i]); else for( size_t i = 0; i < total; i++ ) dst[i] = std::min(src1[i], src2[i]); } static void minmax(const Mat& src1, const Mat& src2, Mat& dst, char op) { dst.create(src1.dims, src1.size, src1.type()); CV_Assert( src1.type() == src2.type() && src1.size == src2.size ); const Mat *arrays[]={&src1, &src2, &dst, 0}; Mat planes[3]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].channels(); size_t i, nplanes = it.nplanes, depth = src1.depth(); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); uchar* dptr = planes[2].ptr(); switch( depth ) { case CV_8U: minmax_((const uchar*)sptr1, (const uchar*)sptr2, (uchar*)dptr, total, op); break; case CV_8S: minmax_((const schar*)sptr1, (const schar*)sptr2, (schar*)dptr, total, op); break; case CV_16U: minmax_((const ushort*)sptr1, (const ushort*)sptr2, (ushort*)dptr, total, op); break; case CV_16S: minmax_((const short*)sptr1, (const short*)sptr2, (short*)dptr, total, op); break; case CV_32S: minmax_((const int*)sptr1, (const int*)sptr2, (int*)dptr, total, op); break; case CV_32F: minmax_((const float*)sptr1, (const float*)sptr2, (float*)dptr, total, op); break; case CV_64F: minmax_((const double*)sptr1, (const double*)sptr2, (double*)dptr, total, op); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } } } void min(const Mat& src1, const Mat& src2, Mat& dst) { minmax( src1, src2, dst, 'm' ); } void max(const Mat& src1, const Mat& src2, Mat& dst) { minmax( src1, src2, dst, 'M' ); } template static void minmax_(const _Tp* src1, _Tp val, _Tp* dst, size_t total, char op) { if( op == 'M' ) for( size_t i = 0; i < total; i++ ) dst[i] = std::max(src1[i], val); else for( size_t i = 0; i < total; i++ ) dst[i] = std::min(src1[i], val); } static void minmax(const Mat& src1, double val, Mat& dst, char op) { dst.create(src1.dims, src1.size, src1.type()); const Mat *arrays[]={&src1, &dst, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total()*planes[0].channels(); size_t i, nplanes = it.nplanes, depth = src1.depth(); int ival = saturate_cast(val); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr1 = planes[0].ptr(); uchar* dptr = planes[1].ptr(); switch( depth ) { case CV_8U: minmax_((const uchar*)sptr1, saturate_cast(ival), (uchar*)dptr, total, op); break; case CV_8S: minmax_((const schar*)sptr1, saturate_cast(ival), (schar*)dptr, total, op); break; case CV_16U: minmax_((const ushort*)sptr1, saturate_cast(ival), (ushort*)dptr, total, op); break; case CV_16S: minmax_((const short*)sptr1, saturate_cast(ival), (short*)dptr, total, op); break; case CV_32S: minmax_((const int*)sptr1, saturate_cast(ival), (int*)dptr, total, op); break; case CV_32F: minmax_((const float*)sptr1, saturate_cast(val), (float*)dptr, total, op); break; case CV_64F: minmax_((const double*)sptr1, saturate_cast(val), (double*)dptr, total, op); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } } } void min(const Mat& src1, double val, Mat& dst) { minmax( src1, val, dst, 'm' ); } void max(const Mat& src1, double val, Mat& dst) { minmax( src1, val, dst, 'M' ); } template static void muldiv_(const _Tp* src1, const _Tp* src2, _Tp* dst, size_t total, double scale, char op) { if( op == '*' ) for( size_t i = 0; i < total; i++ ) dst[i] = saturate_cast<_Tp>((scale*src1[i])*src2[i]); else if( src1 ) for( size_t i = 0; i < total; i++ ) dst[i] = src2[i] ? saturate_cast<_Tp>((scale*src1[i])/src2[i]) : 0; else for( size_t i = 0; i < total; i++ ) dst[i] = src2[i] ? saturate_cast<_Tp>(scale/src2[i]) : 0; } static void muldiv(const Mat& src1, const Mat& src2, Mat& dst, double scale, char op) { dst.create(src2.dims, src2.size, src2.type()); CV_Assert( src1.empty() || (src1.type() == src2.type() && src1.size == src2.size) ); const Mat *arrays[]={&src1, &src2, &dst, 0}; Mat planes[3]; NAryMatIterator it(arrays, planes); size_t total = planes[1].total()*planes[1].channels(); size_t i, nplanes = it.nplanes, depth = src2.depth(); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr1 = planes[0].ptr(); const uchar* sptr2 = planes[1].ptr(); uchar* dptr = planes[2].ptr(); switch( depth ) { case CV_8U: muldiv_((const uchar*)sptr1, (const uchar*)sptr2, (uchar*)dptr, total, scale, op); break; case CV_8S: muldiv_((const schar*)sptr1, (const schar*)sptr2, (schar*)dptr, total, scale, op); break; case CV_16U: muldiv_((const ushort*)sptr1, (const ushort*)sptr2, (ushort*)dptr, total, scale, op); break; case CV_16S: muldiv_((const short*)sptr1, (const short*)sptr2, (short*)dptr, total, scale, op); break; case CV_32S: muldiv_((const int*)sptr1, (const int*)sptr2, (int*)dptr, total, scale, op); break; case CV_32F: muldiv_((const float*)sptr1, (const float*)sptr2, (float*)dptr, total, scale, op); break; case CV_64F: muldiv_((const double*)sptr1, (const double*)sptr2, (double*)dptr, total, scale, op); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } } } void multiply(const Mat& src1, const Mat& src2, Mat& dst, double scale) { muldiv( src1, src2, dst, scale, '*' ); } void divide(const Mat& src1, const Mat& src2, Mat& dst, double scale) { muldiv( src1, src2, dst, scale, '/' ); } template static void mean_(const _Tp* src, const uchar* mask, size_t total, int cn, Scalar& sum, int& nz) { if( !mask ) { nz += (int)total; total *= cn; for( size_t i = 0; i < total; i += cn ) { for( int c = 0; c < cn; c++ ) sum[c] += src[i + c]; } } else { for( size_t i = 0; i < total; i++ ) if( mask[i] ) { nz++; for( int c = 0; c < cn; c++ ) sum[c] += src[i*cn + c]; } } } Scalar mean(const Mat& src, const Mat& mask) { CV_Assert(mask.empty() || (mask.type() == CV_8U && mask.size == src.size)); Scalar sum; int nz = 0; const Mat *arrays[]={&src, &mask, 0}; Mat planes[2]; NAryMatIterator it(arrays, planes); size_t total = planes[0].total(); size_t i, nplanes = it.nplanes; int depth = src.depth(), cn = src.channels(); for( i = 0; i < nplanes; i++, ++it ) { const uchar* sptr = planes[0].ptr(); const uchar* mptr = planes[1].ptr(); switch( depth ) { case CV_8U: mean_((const uchar*)sptr, mptr, total, cn, sum, nz); break; case CV_8S: mean_((const schar*)sptr, mptr, total, cn, sum, nz); break; case CV_16U: mean_((const ushort*)sptr, mptr, total, cn, sum, nz); break; case CV_16S: mean_((const short*)sptr, mptr, total, cn, sum, nz); break; case CV_32S: mean_((const int*)sptr, mptr, total, cn, sum, nz); break; case CV_32F: mean_((const float*)sptr, mptr, total, cn, sum, nz); break; case CV_64F: mean_((const double*)sptr, mptr, total, cn, sum, nz); break; default: CV_Error(Error::StsUnsupportedFormat, ""); } } return sum * (1./std::max(nz, 1)); } void patchZeros( Mat& mat, double level ) { int j, ncols = mat.cols * mat.channels(); int depth = mat.depth(); CV_Assert( depth == CV_32F || depth == CV_64F ); for( int i = 0; i < mat.rows; i++ ) { if( depth == CV_32F ) { float* data = mat.ptr(i); for( j = 0; j < ncols; j++ ) if( fabs(data[j]) < level ) data[j] += 1; } else { double* data = mat.ptr(i); for( j = 0; j < ncols; j++ ) if( fabs(data[j]) < level ) data[j] += 1; } } } static void calcSobelKernel1D( int order, int _aperture_size, int size, vector& kernel ) { int i, j, oldval, newval; kernel.resize(size + 1); if( _aperture_size < 0 ) { static const int scharr[] = { 3, 10, 3, -1, 0, 1 }; assert( size == 3 ); for( i = 0; i < size; i++ ) kernel[i] = scharr[order*3 + i]; return; } for( i = 1; i <= size; i++ ) kernel[i] = 0; kernel[0] = 1; for( i = 0; i < size - order - 1; i++ ) { oldval = kernel[0]; for( j = 1; j <= size; j++ ) { newval = kernel[j] + kernel[j-1]; kernel[j-1] = oldval; oldval = newval; } } for( i = 0; i < order; i++ ) { oldval = -kernel[0]; for( j = 1; j <= size; j++ ) { newval = kernel[j-1] - kernel[j]; kernel[j-1] = oldval; oldval = newval; } } } Mat calcSobelKernel2D( int dx, int dy, int _aperture_size, int origin ) { CV_Assert( (_aperture_size == -1 || (_aperture_size >= 1 && _aperture_size % 2 == 1)) && dx >= 0 && dy >= 0 && dx + dy <= 3 ); Size ksize = _aperture_size == -1 ? Size(3,3) : _aperture_size > 1 ? Size(_aperture_size, _aperture_size) : dx > 0 ? Size(3, 1) : Size(1, 3); Mat kernel(ksize, CV_32F); vector kx, ky; calcSobelKernel1D( dx, _aperture_size, ksize.width, kx ); calcSobelKernel1D( dy, _aperture_size, ksize.height, ky ); for( int i = 0; i < kernel.rows; i++ ) { float ay = (float)ky[i]*(origin && (dy & 1) ? -1 : 1); for( int j = 0; j < kernel.cols; j++ ) kernel.at(i, j) = kx[j]*ay; } return kernel; } Mat calcLaplaceKernel2D( int aperture_size ) { int ksize = aperture_size == 1 ? 3 : aperture_size; Mat kernel(ksize, ksize, CV_32F); vector kx, ky; calcSobelKernel1D( 2, aperture_size, ksize, kx ); if( aperture_size > 1 ) calcSobelKernel1D( 0, aperture_size, ksize, ky ); else { ky.resize(3); ky[0] = ky[2] = 0; ky[1] = 1; } for( int i = 0; i < ksize; i++ ) for( int j = 0; j < ksize; j++ ) kernel.at(i, j) = (float)(kx[j]*ky[i] + kx[i]*ky[j]); return kernel; } void initUndistortMap( const Mat& _a0, const Mat& _k0, Size sz, Mat& _mapx, Mat& _mapy ) { _mapx.create(sz, CV_32F); _mapy.create(sz, CV_32F); double a[9], k[5]={0,0,0,0,0}; Mat _a(3, 3, CV_64F, a); Mat _k(_k0.rows,_k0.cols, CV_MAKETYPE(CV_64F,_k0.channels()),k); double fx, fy, cx, cy, ifx, ify, cxn, cyn; _a0.convertTo(_a, CV_64F); _k0.convertTo(_k, CV_64F); fx = a[0]; fy = a[4]; cx = a[2]; cy = a[5]; ifx = 1./fx; ify = 1./fy; cxn = cx; cyn = cy; for( int v = 0; v < sz.height; v++ ) { for( int u = 0; u < sz.width; u++ ) { double x = (u - cxn)*ifx; double y = (v - cyn)*ify; double x2 = x*x, y2 = y*y; double r2 = x2 + y2; double cdist = 1 + (k[0] + (k[1] + k[4]*r2)*r2)*r2; double x1 = x*cdist + k[2]*2*x*y + k[3]*(r2 + 2*x2); double y1 = y*cdist + k[3]*2*x*y + k[2]*(r2 + 2*y2); _mapy.at(v, u) = (float)(y1*fy + cy); _mapx.at(v, u) = (float)(x1*fx + cx); } } } std::ostream& operator << (std::ostream& out, const MatInfo& m) { if( !m.m || m.m->empty() ) out << ""; else { static const char* depthstr[] = {"8u", "8s", "16u", "16s", "32s", "32f", "64f", "?"}; out << depthstr[m.m->depth()] << "C" << m.m->channels() << " " << m.m->dims << "-dim ("; for( int i = 0; i < m.m->dims; i++ ) out << m.m->size[i] << (i < m.m->dims-1 ? " x " : ")"); } return out; } static Mat getSubArray(const Mat& m, int border, vector& ofs0, vector& ofs) { ofs.resize(ofs0.size()); if( border < 0 ) { std::copy(ofs0.begin(), ofs0.end(), ofs.begin()); return m; } int i, d = m.dims; CV_Assert(d == (int)ofs.size()); vector r(d); for( i = 0; i < d; i++ ) { r[i].start = std::max(0, ofs0[i] - border); r[i].end = std::min(ofs0[i] + 1 + border, m.size[i]); ofs[i] = std::min(ofs0[i], border); } return m(&r[0]); } template static void writeElems(std::ostream& out, const void* data, int nelems, int starpos) { for(int i = 0; i < nelems; i++) { if( i == starpos ) out << "*"; out << (_WTp)((_Tp*)data)[i]; if( i == starpos ) out << "*"; out << (i+1 < nelems ? ", " : ""); } } static void writeElems(std::ostream& out, const void* data, int nelems, int depth, int starpos) { if(depth == CV_8U) writeElems(out, data, nelems, starpos); else if(depth == CV_8S) writeElems(out, data, nelems, starpos); else if(depth == CV_16U) writeElems(out, data, nelems, starpos); else if(depth == CV_16S) writeElems(out, data, nelems, starpos); else if(depth == CV_32S) writeElems(out, data, nelems, starpos); else if(depth == CV_32F) { std::streamsize pp = out.precision(); out.precision(8); writeElems(out, data, nelems, starpos); out.precision(pp); } else if(depth == CV_64F) { std::streamsize pp = out.precision(); out.precision(16); writeElems(out, data, nelems, starpos); out.precision(pp); } else CV_Error(Error::StsUnsupportedFormat, ""); } struct MatPart { MatPart(const Mat& _m, const vector* _loc) : m(&_m), loc(_loc) {} const Mat* m; const vector* loc; }; static std::ostream& operator << (std::ostream& out, const MatPart& m) { CV_Assert( !m.loc || ((int)m.loc->size() == m.m->dims && m.m->dims <= 2) ); if( !m.loc ) out << *m.m; else { int i, depth = m.m->depth(), cn = m.m->channels(), width = m.m->cols*cn; for( i = 0; i < m.m->rows; i++ ) { writeElems(out, m.m->ptr(i), width, depth, i == (*m.loc)[0] ? (*m.loc)[1] : -1); out << (i < m.m->rows-1 ? ";\n" : ""); } } return out; } MatComparator::MatComparator(double _maxdiff, int _context) : maxdiff(_maxdiff), realmaxdiff(DBL_MAX), context(_context) {} ::testing::AssertionResult MatComparator::operator()(const char* expr1, const char* expr2, const Mat& m1, const Mat& m2) { if( m1.type() != m2.type() || m1.size != m2.size ) return ::testing::AssertionFailure() << "The reference and the actual output arrays have different type or size:\n" << expr1 << " ~ " << MatInfo(m1) << "\n" << expr2 << " ~ " << MatInfo(m2) << "\n"; //bool ok = cvtest::cmpUlps(m1, m2, maxdiff, &realmaxdiff, &loc0); int code = cmpEps( m1, m2, &realmaxdiff, maxdiff, &loc0, true); if(code >= 0) return ::testing::AssertionSuccess(); Mat m[] = {m1.reshape(1,0), m2.reshape(1,0)}; int dims = m[0].dims; vector loc; int border = dims <= 2 ? context : 0; Mat m1part, m2part; if( border == 0 ) { loc = loc0; m1part = Mat(1, 1, m[0].depth(), m[0].ptr(&loc[0])); m2part = Mat(1, 1, m[1].depth(), m[1].ptr(&loc[0])); } else { m1part = getSubArray(m[0], border, loc0, loc); m2part = getSubArray(m[1], border, loc0, loc); } return ::testing::AssertionFailure() << "too big relative difference (" << realmaxdiff << " > " << maxdiff << ") between " << MatInfo(m1) << " '" << expr1 << "' and '" << expr2 << "' at " << Mat(loc0).t() << ".\n" << "- " << expr1 << ":\n" << MatPart(m1part, border > 0 ? &loc : 0) << ".\n" << "- " << expr2 << ":\n" << MatPart(m2part, border > 0 ? &loc : 0) << ".\n"; } void threshold( const Mat& _src, Mat& _dst, double thresh, double maxval, int thresh_type ) { int i, j; int depth = _src.depth(), cn = _src.channels(); int width_n = _src.cols*cn, height = _src.rows; int ithresh = cvFloor(thresh); int imaxval, ithresh2; if( depth == CV_8U ) { ithresh2 = saturate_cast(ithresh); imaxval = saturate_cast(maxval); } else if( depth == CV_16S ) { ithresh2 = saturate_cast(ithresh); imaxval = saturate_cast(maxval); } else { ithresh2 = cvRound(ithresh); imaxval = cvRound(maxval); } assert( depth == CV_8U || depth == CV_16S || depth == CV_32F ); switch( thresh_type ) { case CV_THRESH_BINARY: for( i = 0; i < height; i++ ) { if( depth == CV_8U ) { const uchar* src = _src.ptr(i); uchar* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) dst[j] = (uchar)(src[j] > ithresh ? imaxval : 0); } else if( depth == CV_16S ) { const short* src = _src.ptr(i); short* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) dst[j] = (short)(src[j] > ithresh ? imaxval : 0); } else { const float* src = _src.ptr(i); float* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) dst[j] = (float)((double)src[j] > thresh ? maxval : 0.f); } } break; case CV_THRESH_BINARY_INV: for( i = 0; i < height; i++ ) { if( depth == CV_8U ) { const uchar* src = _src.ptr(i); uchar* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) dst[j] = (uchar)(src[j] > ithresh ? 0 : imaxval); } else if( depth == CV_16S ) { const short* src = _src.ptr(i); short* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) dst[j] = (short)(src[j] > ithresh ? 0 : imaxval); } else { const float* src = _src.ptr(i); float* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) dst[j] = (float)((double)src[j] > thresh ? 0.f : maxval); } } break; case CV_THRESH_TRUNC: for( i = 0; i < height; i++ ) { if( depth == CV_8U ) { const uchar* src = _src.ptr(i); uchar* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { int s = src[j]; dst[j] = (uchar)(s > ithresh ? ithresh2 : s); } } else if( depth == CV_16S ) { const short* src = _src.ptr(i); short* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { int s = src[j]; dst[j] = (short)(s > ithresh ? ithresh2 : s); } } else { const float* src = _src.ptr(i); float* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { double s = src[j]; dst[j] = (float)(s > thresh ? thresh : s); } } } break; case CV_THRESH_TOZERO: for( i = 0; i < height; i++ ) { if( depth == CV_8U ) { const uchar* src = _src.ptr(i); uchar* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { int s = src[j]; dst[j] = (uchar)(s > ithresh ? s : 0); } } else if( depth == CV_16S ) { const short* src = _src.ptr(i); short* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { int s = src[j]; dst[j] = (short)(s > ithresh ? s : 0); } } else { const float* src = _src.ptr(i); float* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { float s = src[j]; dst[j] = s > thresh ? s : 0.f; } } } break; case CV_THRESH_TOZERO_INV: for( i = 0; i < height; i++ ) { if( depth == CV_8U ) { const uchar* src = _src.ptr(i); uchar* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { int s = src[j]; dst[j] = (uchar)(s > ithresh ? 0 : s); } } else if( depth == CV_16S ) { const short* src = _src.ptr(i); short* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { int s = src[j]; dst[j] = (short)(s > ithresh ? 0 : s); } } else { const float* src = _src.ptr(i); float* dst = _dst.ptr(i); for( j = 0; j < width_n; j++ ) { float s = src[j]; dst[j] = s > thresh ? 0.f : s; } } } break; default: assert(0); } } static void _minMaxIdx( const float* src, const uchar* mask, double* _minVal, double* _maxVal, size_t* _minIdx, size_t* _maxIdx, int len, size_t startIdx ) { double minVal = FLT_MAX, maxVal = -FLT_MAX; size_t minIdx = 0, maxIdx = 0; if( !mask ) { for( int i = 0; i < len; i++ ) { float val = src[i]; if( val < minVal ) { minVal = val; minIdx = startIdx + i; } if( val > maxVal ) { maxVal = val; maxIdx = startIdx + i; } } } else { for( int i = 0; i < len; i++ ) { float val = src[i]; if( mask[i] && val < minVal ) { minVal = val; minIdx = startIdx + i; } if( mask[i] && val > maxVal ) { maxVal = val; maxIdx = startIdx + i; } } } if (_minIdx) *_minIdx = minIdx; if (_maxIdx) *_maxIdx = maxIdx; if (_minVal) *_minVal = minVal; if (_maxVal) *_maxVal = maxVal; } void minMaxIdx( InputArray _img, double* minVal, double* maxVal, Point* minLoc, Point* maxLoc, InputArray _mask ) { Mat img = _img.getMat(); Mat mask = _mask.getMat(); CV_Assert(img.dims <= 2); _minMaxIdx((const float*)img.data, mask.data, minVal, maxVal, (size_t*)minLoc, (size_t*)maxLoc, (int)img.total(),1); if( minLoc ) std::swap(minLoc->x, minLoc->y); if( maxLoc ) std::swap(maxLoc->x, maxLoc->y); } }